以 含金量 勇敢 挑战 国内 外 同类 书籍 


除 正 沖 ”编著 
石 虎 ”审阅 




















用 试 笔试 的 秘密 


和 版权 申明 


本 书 尚未 出 版 , 先 放 到 网 上 给 大 家 免费 
下 载 和 阅览 。 本 书 正式 出 版 前 读者 可 以 仔细 
研读 和 目 由 传阅 本 书 电子 版 , 但 不 允许 私 目 
大 量 印 刷 和 销售 。 出 版 社 如 想 出 版 此 书 可 通 
过 邮件 或 博客 留言 联系 作者 商谈 出 版 事宜 。 

对 于 非法 盗 印 或 盗版 , 作者 閣 本 看 愚 公 
移 山 的 精神 ， 和 孜孜 不 倦 的 与 盗版 者 周旋 ， 直 
至 法 律 做 出 公正 的 裁决 。 


写 在 前 言 前 面 的 话 

最 近 面 试 了 一 些 人 ， 包 括 应 届 本 科 、 硕 士 和 工作 多 年 的 程序 员 ， 
在 问 到 C 语言 相关 的 问题 的 时 候 ， 总 是 没 几 个 人 能 完全 答 上 我 的 问 
题 。 甚 至 一 些 工 作 多 年 ， 简 历 上 写 着 “最 得 意 的 语言 是 C 语言 “对 
C 有 很 深 的 研究 “精通 C 语言 ”的 人 也 答 不 完全 我 的 问题 ， 甚 至 有 
个 别人 我 问 的 问题 一 个 都 答 不 上 。 于 是 我 就 想起 了 我 去 年 内 的 使 用 写 
的 这 本 小 册子 。 

这 本 小 册子 已 经 在 我 电脑 里 睡 了 一 年 大 觉 了 。 并非 没有 出 版 社 愿 
意 出 版 , 而 是 几 个 大 的 出 版 社 都 认为 书写 得 不 错 , 但 太 薄 , 利润 太 低 ， 
所 以 要 求 我 加 厚 到 300 页 以 上 。 我 拒绝 加 厚 ， 并 为 此 和 几 个 出 版 社 伪 
持 了 一 年 多 。 我 认为 经 典 的 东西 一 定 要 精炼 ， 不 要 废话 。 这 次 由 于 面 
试 别 人 ， 所 以 终于 记 起 了 我 还 写 过 这 么 一 本 小 册子 。 想 了 想 , 还 是 决 
定 挂 到 网 上 免费 让 大 家 看 得 了 。 并 为 此 专门 为 本 书 开 了 个 博客 ,以 方 








便 和 读者 交流 。 博 客 地 址 : http://blog. csdn. net/dissection c 

作者 简介 : 

陈 正 冲 : 湖南 沅江 人 ， 毕 业 于 长 春光 学 精密 机 械 学 院 〈 长 春 理工 
大 学 ) 数学 系 。 目 前 从 事 舱 入 式 软件 开发 和 管理 方面 的 工作 。 

石 虎 : 湖南 沅江 人 人， 毕业 于 吉林 大 学 计算 机 系 。 目 前 为 大 连 交 通 
大 学 计算 机 系 讲 师 。 





我 遇 到 过 很 多 程序 员 和 计 
解 过 《高 级 C 语言 程序 设计 》。 每 期 班 开课 前 ， 我 总 会 问 学 生 : 你 感觉 C 语言 学 得 怎么 样 ? 
吗 ? 数组 呢 ? 内 存 管理 
针 很 明白 ， 数 组 很 简单 ， 内 存 管 理 也 不 难 。 
你 想 达到 什么 程度 ? 很 多 学 生 回答 : 精通 C 语言 。 
为 我 完全 在 和 一 群 业余 者 或 者 是 C 语言 爱好 者 在 对 话 。 你 们 大 学 的 计算 机 教育 根本 就 是 在 
浪费 你 们 的 时 间 ， 念 了 几 年 大 学 ， 连 C 语言 的 门 都 没 摸 着 。 现 在 大 多 数学 校 计算 机 系 都 开 
好 像 什么 都 学 了 ， 但 是 什么 都 不 会 ， 更 可 悲 的 是 有 些 大 学 


























了 C、C++、Java、C# 等 




















机 系 毕 业 的 





难 吗 ? 指针 明 



































前 
学 4 








lll 


E, 也 给 很 多 程序 员 和 计算 机 系 毕 业 的 学 生 ; 












































I? 往往 学 生 








回答 说 : 感觉 还 可 以 , C 语言 不 难 , 指 























般 我 会 再 问 一 个 问题 ， 通 过 这 个 班 的 学 习 ， 











告诉 他 们 : 我 很 无 条 ， 也 很 无 语 。 因 









































居然 取消 了 C 语言 课程 ， 认 为 其 过 时 了 。 我 个 人 的 观点 是 “十 乌 在 林 ， 不 如 一 乌 在 手 ”， 真 








EJE C 语言 整 明 

























































































没 掌握 C 语言 ，5 年 之 下 ， 一 般 来 说 还 没 训 
告诉 我 的 学 生 : 听 完 我 的 课 ， 远 达 不 到 精通 的 目标 ， 熟 悉 也 达 不 到 ， 掌 握 也 达 不 到 。 那 能 
入 C 语 言 的 大 门 。 入 门 之 后 的 造化 如 何在 于 你 们 自己 。 不 过 我 
EREM F10 或 F11 按 坏 ， 当 然 不 能 是 垃圾 键盘 。 
着 疑虑 。C 语言 有 这 么 难 吗 ? 我 的 回答 是 : 不 难 。 但 





达到 什么 目标 ? 
可 以 告诉 你 们 一 条 不 是 扣 
往往 讲 到 这 里 ， 学 4 












































了 再 学 别 的 语言 也 很 简单 ， 如 果 C 语言 都 没 整 明白 ， 别 的 语言 学 得 再 好 














怎么 回 事 。 当 然 我 也 从 来 不 认为 一 个 没 学 过 汇编 的 人 

















能 真正 掌握 C 语言 的 真 诺 。 我 个 人 一 直 认 为 ， 普 通 人 用 C 语言 在 3 年 之 下 ， 一 般 来 说 ， 还 















































E 径 的 捷径 :把 一 个 鲁 
We 


VIER 


























你 就 是 用 不 明白 。 学 生 说 ， 以 前 大 学 老 曙 
试 也 很 好 。 平 时 练习 感觉 自 
ET, KETER 
表 你 真正 懂 了 ! 什么 时 候 表 明 


白 不 代表 你 



































`F = 
ACER; 





10 年 之 下 ， 谈 不 上 精通 。 所 以 ， 我 





















































者 C 语言 ， 我 学 得 很 好 。 老 师 讲 的 都 能 听 懂 ， 考 
， 工 作 也 很 轻松 找到 了 。 我 告诉 学 生 : 听 明 白 ， 看 明 

























































































学 生 都 听 明 






















































































冰山 大 家 都 没 见 过 ， 但 总 听 过 或 是 
也 算 个 人 物 G 
里 工 科 的 ， 应 该 明白 冰山 在 水 本 












































坦 尼克 》 里 的 冰山 给 泰坦 尼克 造成 了 巨大 的 损失 。 你 们 都 是 
的 部 分 只 是 总 个 冰山 的 /8。 我 现在 就 告诉 你 们 ，C 语言 








就 是 这 座 冰 山 。 你 们 现在 仅仅 
希望 通过 我 的 讲解 ， 让 你 们 摸 到 水 面 下 的 部 分 ， 


从 现在 开始 ， 除 非 在 特殊 








4 况 下， 不 允许 月 



































了， 你 会 用 了 不 代表 你 能 用 明白 ， 你 能 用 明白 不 代 
FE 懂 了 呢 ? 你 站 在 我 这 来 ， 把 问题 给 下 面 的 同学 讲 明 白 ， 


不 


舍 则 ， 你 就 没 真正 懂 ， 这 是 检验 懂 没 慌 的 唯一 标准 。 













































































BZ R A PH? 如 果 你 连 《泰坦 尼 死 》 都 没 看 过 ， 那 你 













































































摸 到 了 水 面 上 的 部 分 ， 甚 至 根本 不 知道 水 面 下 的 部 分 。 我 
让 你 们 知道 C 语言 到 底 是 什么 样子 。 


H printf 这 个 函数 。 为 什么 呢 ? 很 多 学 生 写 完 



































代码 ， 直 接 用 printf 打印 出 来 ， 发 现 结果 不 对 。 然 后 就 举 手 问 我 :老师 ， 我 的 结果 为 什么 不 








多 时 候 printf 出 来 的 结果 是 对 的 , 然后 
往往 不 是 ， 往 后 看 
有 问题 。 所 以 ， 


























对 啊 ? 连 调试 的 意识 都 没有 ! 大 多 数学 生根 本 就 不 会 调试 ， 不 会 看 变量 的 值 ， 内 存 的 值 。 














只 知道 printf 出 来 结果 不 对 ， 却 不 知 





道 为 什么 不 对 ， 


















































前 的 编译 器 上 


看 到 例子 的 。 





这 个 时 候 呢 ， 























妙 。 
























































怎么 解决 。 这 种 情况 还 算 好 的 。 往 往 很 
呢 , 学 生 也 理所当然 的 认为 程序 没有 问题 。 是 这 样 吗 ? 
到 例子 的 。 永 远 给 我 记 住 一 点 : 结果 对 ， 并 不 代表 程序 真正 没 





























尽量 不 要 用 printf 函数 ， 要 去 看 变量 的 值 ， 内 存 的 值 。 当 然 ， 在 我 们 目 
， 变 量 的 值 ， 内 存 的 值 对 了 就 代表 你 程序 没 问 题 吗 ? 也 不 是 ， 往 后 ， 你 也 会 



























































这 个 老师 有 问题 吧 。 大 学 里 我 们 老师 都 教 我 们 怎么 





用 printf， 告 诉 我 们 要 经 常用 printf。 这 也 恰恰 是 大 学 教育 失败 的 地 方 之 一 。 很 多 大 学 老师 根 














本 就 没 真正 用 C 语言 写 过 几 行 代码 , 更 别 说 教学 生 j 
水 平 永远 也 无 法 提 上 来 ， 所 以 ， 要 想 学 好 一 门 编程 



































周 试 代码 了 ,不 调试 代码 ,不 按 F10 或 F11, 

















语言 ， 最 好 的 办 法 就 是 多 调试 。 你 去 一 





个 软件 公司 转 转 ， 去 看 人 家 的 键盘 ， 如 果 发 现 键盘 上 的 F10 或 F11 铮 亮 铮 亮 ， 毫 无 疑问 ， 


此 机 的 主人 曾经 或 现 帮 








E 是 开发 人 员 (这 





























仅 指 写 代 码 的 , 不 上 升 到 架构 设计 类 的 开发 人 员 )， 


不 
==) 


则 ， 














比较 擅长 。 
当 则 是 见 

















讲解 的 时 候 会 举 


二 见 智 


必 是 非 开 发 人 员 。 
非常 有 必要 申明 ,本 人 并 非 什 么 学 者 或 是 专家 , 但 本 人 是 数学 系 毕 业 ， 所 以 对 理 
白 这 个 知识 点 ， 至 于 这 些 例子 
练 使 得 本 人 思 














知 识 点 尤其 是 


些 概念 性 原理 性 








的 。 本 书 是 我 3 


多 期 培训 班 的 实践 ， 
我 有 生 以 来 听课 从 来 都 没有 听 得 这 人 么 











时 





解 C 语言 的 一 些 心得 
































的 东西 





民 多 例子 来 尽量 使 学 生 明 
的 问题 了 。 但 是 一 条 ， 长 期 的 数学 训 
时 会 抠 的 很 细 、 
和 经 验 ， 





















































RIXE 

















工作 来 听 我 


的 课 的 。 
当然 ， 关 于 C 语言 的 这 么 多 经 验 和 心得 











不 过 是 站 在 








解 得 比较 透彻 ， 
透彻 ， 这 人 么 








FA BJ) ET = o 
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维 比较 严谨 ， 














W JI 
是 否 恰 


解 一 些 
































讲 


民 严 ， 这 一 点 相信 读者 会 体会 得 到 
中 有 很 多 我 个 人 的 见解 或 看 法 。 


经 过 
と 





学生 所 得 


的 积累 


明白 。 





很 多 学 生 听 完 课 
明白 过 。 也 有 业余 班 的 学 生 甚 至 辞 

















后 告诉 我 : 


掉 本 职 























并非 我 一 人 之 力 。 借 用 
给 学 生 做 培训 的 时 候 我 参考 得 比较 多 的 书 有 : 

















Ritchie 的 《The C Programming Language》; Linden 的 《Expert C Programming》; 
Koening (C Traps and Pitfalls}; 


(Code Complete. Second Edition}; 
但 却 都 有 着 各 自 


作 * 


的 缺陷 。 




















我 的 讲 














以 ， 至 少 还 没有 出 乱 子 。 





者 能 读 上 十 遍 。 











见解 深刻 ， 使 我 受益 





HIF, TR 
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平 也 仅仅 是 入 门 而 已 。 


学 习 C 语言 ， 这 几 本 书 丸 
， 不 如 说 本 书 是 我 对 这 


验 与 心得 

















基础 知识 ， 所 以 ， 本 书 并 不 适用 

















大 的 计 
毕业 生 ? P JÉ 

















机 系 的 学 生 和 初级 程序 员 。 
没有 


家 大 公司 会 














级 程序 员 参 考 。 


说 是 件 非常 好 的 事情 。 有 人 说 | 
的 发 现 一 些 本 来 可 以 做 得 
些 地 方 或 是 没有 











尤其 





Steve Maguire 的 《Write Clean Code}; 
林 锐 的 《高 质量 C++/C 编程 指南 》。 
读 这 些 书 才能 深刻 的 千 握 茶 知识 点 。 





读者 往往 需要 同时 阅 
课 的 试图 时 候 融 各 家 之 长 ， 再 加 上 我 个 人 的 见解 传授 给 学 生 。 
这 些 书 饱 含 着 作者 的 箱 
写本 书 时 也 参考 了 网 上 一 些 无 名 高 手 的 文章 ， 这 
浅 。 这 里 要 感谢 这 些 大 师 们 ， 如 果 不 是 他 们 ， 肯 怕 我 的 C 语言 的 水 


=P 
EEN 





[果真 正 哺 透 了 ， 水 平 不 会 差 到 哪 。 











每 读 一 








些 大 师 们 智 意 的 解读 。 本 了 
语言 零 基 而 
讲 的 深 的 多 ， 其 中 有 很 多 问题 是 各 大 公司 的 面试 或 笔 








于 C 





























上 的 人 。 本 书 的 知识 要 比 
试题 。 所 以 本 书 的 读者 应 该 





拒绝 你 。 





























已 








更 好 的 
或 是 没 能 








到 ， 




















也 是 ， 为 了 尽量 


EE 精炼 ， 总 是 犹 耶 一 些 








E1⁄ Ei 
ZIE 
缺陷 。 
讲 透 




















当然 ， 


书 内 和 


如 果 本 书 上 面 的 问题 能 真正 明 











书 内 的 一 些 例子 或 比方 ， 如 果 能 被 广大 教 
门 遗憾 的 艺术 ， 
讲课 同样 


也 如 此 ， 每 次 ; 
彻 或 是 忘 了 举 一 个 轻 浅 的 侦 


与 其 


民 多 知识 也 值得 计 





名 名 


我 只 


HO: 


Kernighan & 


Andrew & 


SIEVE McConnell 的 
这 些 书 都 





=} 


是 经 典 之 














还 好 ， 


文 些 高 








学 生 反 映 还 可 





饥 都 有 不 同 的 收获 ， 我 希望 读 


高 手 的 文章 


说 本 书 是 我 授课 的 经 
并 不 是 从 头 到 尾 讲解 C 语言 的 
一 般 的 C 语言 书 说 











白 80%, 


作为 一 个 应 














是 中 国 


J 








= 
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机 教师 














IHJ 








课堂 ? 我 想 


或 是 中 高 
对 学 生来 











因为 在 





; 辑 
































完成 之 后 总 能 




















东西 的 去 留 。 


限于 作者 水 平 ， 


错误 ， 和 希望 各 位 读者 能 予 指教 。 作 者 Mail:dissection_c@163.com. 








完 课 之 后 总 
| 子 等 等 。 整理 本 
书 中 难免 有 些 i 








陈 正 冲 


2008 年 6 





或 多 或 少 
现 自己 某 
书 的 过 程 
遗漏 甚至 

















23 日 
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第 一 章 关键 字 

















每 次 讲 关 键 字 之 前 , 我 总 是 问 学 生 : C 语言 有 多 少 个 关键 字 ? sizeof 怎么 用 ? CERM 











吗 ? 有 些 学 生 不 知 








> 




















JE 


C 语言 有 多 少 个 关键 字 ， 大 多 数学 4 

















它 后 面 跟着 一 对 括号 。 当 投影 仪 把 这 32 个 关键 字 投 到 幕布 上 时 ，4 
关键 字 从 来 没 见 过 ， 有 的 惊讶 C 语言 关键 字 竞 有 32 个 之 多 。 更 有 


















































往往 告诉 我 sizeof 是 函数 ， 因 为 





R 多 学 生 表情 惊讶 。 有 些 





























tf 者， 说 大 学 老师 告诉 他 








们 sizeof 是 函数 ， 没 想到 它 居 然 是 关键 字 ! 由 此 可 想 而 知 ， 大 学 的 计算 机 教育 是 多 么 失败 ! 





int 
double 
long 
char 
float 
short 
signed 
unsigned 
struct 
union 
enum 
static 
switch 
case 
default 
break 
register 
const 
volatile 


typedef 
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TOH EEH 3 H HH He HEH H H 
= 








表 (1.1)C 语言 标准 定义 的 32 个 关键 字 
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= 





整 型 变量 
双 精 度 变量 
长 整 型 变量 
字符 型 变量 
浮 点 型 变量 
] 短 整 型 变量 
有 符号 类 型 变量 
无 符号 类 型 变量 
结构 体 变量 
联合 数据 类 型 
枚 举 类 型 


= 





= 




















hera; 


KF 态 变量 


日 于 开关 语句 








= 








开关 语句 分 文 








开关 语句 中 的 “其 他 ”分 文 
跳出 当前 循环 


声明 寄存 器 变量 


[El 








声明 只 读 变量 


说 明 变 量 在 程序 执行 中 可 被 隐 含 地 改变 


月 





























以 给 数据 类 型 取 别名 (当然 还 有 其 他 作用 
































自动 变量 ， 缺 省 时 编译 器 一 般 默 认为 auto 


extern 
return 
void 


continue 
































声明 变量 是 在 其 他 文件 正 声明 (也 可 以 看 做 是 引用 变量 ) 
子 程序 返回 语句 (可 以 带 参 数 ， 也 可 不 带 参数 ) 

声明 函数 无 返回 值 或 无 参数 ， 声 明 空 类 型 指针 

结束 当前 循环 ， 开 始 下 一 轮 循环 

循环 语句 的 循环 体 
循环 语句 的 循环 条 件 
条 件 语句 

条 件 语句 否定 分 支 ( 与 让 连用 ) 
一 种 循环 语句 (可 意 会 不 可 言传 ) 
无 条 件 跳 转 语句 

计算 对 象 所 占 内 存 空间 大 小 



























































下 面 的 篇 幅 就 一 一 讲解 这 些 关 键 字 。 但 在 讲解 之 前 先 明确 两 个 概念: 












































A)int i; 


B)extern int i; (关于 extern， 后 面 解释 ) 














什么 是 定义 ? 什么 是 声明 ? 它们 有 何 区 别 ? 
2 











哪个 是 定义 ? 哪个 是 声明 ? 或 者 都 是 定义 或 者 都 是 声明 ? 我 所 教 过 的 学 生 几 乎 没有 一 
人 能 回答 上 这 个 问题 。 这 个 十 分 重要 的 概念 在 大 学 里 从 来 没有 被 提起 过 ! 

什么 是 定义 : 所 谓 的 定义 就 是 (编译 器 ) 创 建 一 个 对 象 ， 为 这 个 对 象 分 配 一 块 内 存 并 给 它 
取 上 一 个 名 字 ， 这 个 名 字 就 是 我 们 经 常 所 说 的 变量 名 或 对 象 名 。 但 注意 ， 这 个 名 字 一 旦 和 
这 块 内 存 匹 配 起 来 (可 以 想象 是 这 个 名 字 嫁 给 了 这 块 空间 ， 没 有 要 彩礼 啊 。^^)， 它 们 就 同 








ERWE, AEDADE. 



















































































并 且 这 块 内 存 的 位 置 也 不 能 被 改变 。 一 个 变量 或 对 象 在 一 定 的 区 

















域内 《比如 函数 内 ， 全 局 等 ) 只 能 被 定义 一 次 ， 如 果 定 义 多 次 ， 编 译 器 会 提示 你 重复 定义 


同一 个 变量 或 对 象 。 








什么 是 声明 : 有 两 重 含义 ， 如 下 ; 


第 一 重 含义 : 告诉 编译 器 ， 这 个 名 字 已 经 匹配 到 一 块 内 存 上 了 (伊人 已 嫁 ， 吾 将 何 去 何 
从 ? 何以 解忧 ， 唯 有 稀 粥 )， 下 面 的 代码 用 到 变量 或 对 象 是 在 别 的 地 方 定义 的 。 声 明 可 以 出 








现 多 次 。 









































译 器 ,我 这 个 名 字 我 先 预定 了 ， 别 的 地 方 再 也 不 能 用 它 来 作为 变量 











名 或 对 象 名 。 比 如 你 在 图 书馆 自习 室 的 某 个 座位 上 放 了 一 本 书 ， 表 明 这 个 座位 已 经 有 人 预 


















































W, 别人 再 也 不 允许 使 


j 这 个 座位 。 其 实 这 个 时 候 你 本 人 并 没有 坐 在 这 个 座位 上 。 这 种 声 



































明 最 典型 的 例子 就 是 函数 参数 的 声明 ， 例 如 : 


void fun(int i, char 























c); 








好 ， 这 样 一 解释 ， 我 们 可 以 很 清楚 的 判断 :A) 是 定义 ，B) 是 声明 。 
那 他 们 的 区 别 也 很 清晰 了 。 记 住 ， 定 义 声 明 最 重要 的 区 别 : 定义 创建 了 对 象 并 为 这 个 





対象 分 配 了 内 存 , 声明 没有 分 配 内 存 (一 介抱 伊 人 , 一 全 喝 稀 弟 。^_^)。 


1.1， 最 宽 恒 大 量 的 关键 字 一 --auto 





auto: 它 很 宽 恒 大 量 的， 你 就 当 它 不 存在 吧 。 编 译 器 在 默认 的 缺 省 情况 下 ， 所 有 变量 





都 是 auto 的 。 


1.2， 最 快 的 关键 字 ---- register 














回 





register: 这 个 关键 字 请 求 编 译 器 尽 可 能 的 将 变量 存在 CPU 内 部 寄存 器 中 而 不 是 通过 内 
































存 寻 址 访问 以 提高 效率 。 注 意 是 尽 可 能 ， 不 是 绝对 。 你 








一 个 CPU 的 寄存 器 也 就 那么 

















几 个 或 几 十 个 ， 你 要 是 定义 了 很 多 很 多 register 变量 ， 它 累 死 也 可 能 不 能 全 部 把 这 些 变量 放 





入 寄存 器 吧 ， 轮 也 可 能 轮 不 到 你 。 


1.2.1， 皇 帝 身 边 的 小 太监 一 一 寄存 器 











不 知道 什么 是 寄存 器 ? 那 见 过 太监 没有 ? KA? 其 实 我 也 没有 。 没 见 过 不 要 紧 ， 见 过 就 
麻烦 大 了 。^^， 大 家 都 看 过 古装 戏 ， 那 些 皇 融 们 要 阅读 奏章 的 时 候 ， 大 臣 总 是 先 将 奏章 交 
给 旦 帝 旁 边 的 小 太监 ， 小 太监 呢 再 交 给 皇帝 同志 人 处理 。 这 个 小 太监 只 是 个 中 转 站 ， 并 无 别 


















































的 功能 。 














好 ， 那 我 们 再 联想 到 我 们 的 CPU. CPU 不 就 是 我 们 



































星 帝 同志 么 ? 大 臣 就 相当 于 我 们 





























的 内 存 ， 数 据 从 他 这 拿 出 来 。 那 小 太监 就 是 我 们 的 寄存 器 了 这 里 先 不 考虑 CPU 的 高 速 组 








FX) 数据 从 内 存 里 拿 出 来 先 放 到 寄存 器 , 然后 CPU P 








从 寄存 器 里 读 取 数据 来 处 理 ， 处 理 





完 后 同样 把 数据 通过 寄存 器 存放 到 内 存 里 ，CPU 不 直接 和 内 存 打 交道 。 这 里 要 说 明 的 一 点 












































是 :小 太监 是 主动 的 从 大 臣 手 里 接 过 奏章 ， 然 后 主动 的 交 























同志 ， 但 寄存 器 没 这 么 自觉 ， 





它 从 不 主动 干什么 事 。 一 个 旺 帝 可 能 有 好 些小 太监 ， 那 么 一 个 CPU 也 可 以 有 很 多 寄存 器 ， 























不 同型 号 的 CPU 拥有 寄存 器 的 数量 不 一 样 。 
































为 喻 要 这 么 麻烦 啊 ? 速度 ! 就 是 因为 速度 。 寄 存 器 其 














实 就 是 一 块 一 块 小 的 存储 空间 ， 只 


不 过 其 存 取 速 度 要 比 内 存 快 得 多 。 进 水 楼 台 先 得 月 嘛 ， 它 离 CPU 很 近 ，CPU 一 伸手 就 拿 到 











数据 了 ， 比 在 那么 大 的 一 块 内 存 里 去 寻找 某 个 地 址 上 的 数 和 



































是 不 是 快 多 了 ? 那 有 人 问 既然 














它 速度 那么 快 ， 那 我 们 的 内 存 硬 盘 都 改 成 寄存 器 得 了 呐 。 我 要 说 的 是 :你 真有 钱 ! 


1. 2. 2， 使 用 register 修饰 符 的 注意 点 

















虽然 寄存 器 的 速度 非常 快 ， 但 是 使 用 register 修饰 符 也 有 些 限制 的 : register 变量 必须 是 









































能 被 CPU 寄存 器 所 接受 的 类 型 。 意味 着 register 变量 必须 是 


























个 单个 的 值 , 并 且 其 长 度 应 小 








于 或 等 于 整 型 的 长 度 。 而 且 register 变量 可 能 不 存放 在 内 存 中 , 所 以 不 能 用 取 址 运算 符 “&” 





来 获取 register 变量 的 地 址 。 


3， 最 名 不 符 实 的 关键 字 - 一 static 

















不 要 误 以 为 关键 字 static 很 安静 ， 其 实 它 一 点 也 不 安静 。 这 个 关键 字 在 C 语言 里 主要 有 
两 个 作用 ，C++ 对 它 进 行 了 扩展 。 














1. 3. 1， 修 饰 变量 


















































第 一 个 作用 : 修饰 变量 。 变 量 又 分 为 局 部 和 全 局 变量 ， 但 它们 都 存在 内 存 的 静态 


静态 全 局 变量 , 作用 域 仅 限于 变量 被 定义 的 文件 中 , 其 他 文件 即使 用 extem 声明 也 没 法 
使 用 他 。 准 确 地 说 作用 域 是 从 定义 之 处 开始 ， 到 文件 结尾 处 结束 ， 在 定义 之 处 前 面 的 那些 
代码 行 也 不 能 使 用 它 。 想 要 使 用 就 得 在 前 面 再 加 extern ***。 恶 心 吧 ? 要 想 不 恶 ， 
直接 在 文件 顶端 定义 不 就 得 了 。 





区 | 

























































































EK 心 ， 很 简单 ， 









































静态 局 部 变量 ， 在 函数 体 里 面 定义 的 ， 就 只 能 在 这 个 函数 里 用 了 ， 同 一 个 文档 中 的 其 他 
函数 也 用 不 了 。 由 于 被 static 修饰 的 变量 总 是 存在 内 存 的 静态 区 ， 所 以 即使 这 个 函数 运行 结 
K, KASTE 的 值 还 是 不 会 被 销毁 ， 函 数 下 次 使 用 时 仍然 能 用 到 这 个 值 。 


static int j; 






















































































void fun1 (void) 
{ 
static int i = 0; 
i++; 
} 
void fun2 (void) 


int main() 
{ 
for(k=0; k<10; k++) 
{ 
fun10: 
fun20: 
) 


return 0: 


) 
i 和 j 的 值 分 别 是 什么 ,为 什么 ? 





1. 3. 2， 修 饰 函 数 


第 二 个 作用 : 修饰 函数 。 函 数 前 加 static 使 得 函数 成 为 静态 函数 。 但 此 处 “static” 的 含义 
不 是 指 存储 方式 ， 而 是 指 对 函数 的 作用 域 仅 局 限于 本 文件 (所 以 又 称 内 部 函数 )。 使 用 内 部 函 
数 的 好 处 是 : 不 同 的 人 编写 不 同 的 函数 时 ， 不 用 担心 自己 定义 的 函数 ， 是 否 会 与 其 它 文件 
中 的 函数 同名 。 
关键 字 static 有 着 不 寻常 的 历史 。 起 初 ， 在 C 中 引入 关键 字 static 是 为 了 表示 退出 一 个 
块 后 仍然 存在 的 局 部 变量 。 随 后 ，static 在 C 中 有 了 第 二 种 含义 : 用 来 表示 不 能 被 其 它 文件 
访问 的 全 局 变量 和 函数 。 为 了 避免 引入 新 的 关键 字 ， 所 以 仍 使 用 static 关键 字 来 表示 这 第 二 
种 含义 。 

当然 ，C++ 里 对 static 赋予 了 第 三 个 作用 ， 这 上 
究 。 



































































































































HH 
NY. 








E 不 讨论 ， 有 兴趣 的 可 以 找 相关 资料 











1.4, 基本 数据 类 型 ----short、 int, long, char, float, double 



































短 整 型 short 
C 语言 包含 的 数据 类 型 如 下 图 所 示 : 整 整 型 int 
数值 类 型 型 长 整 型 Jong 
单 精 度 型 float 
基本 类 型 No 双 精 度 型 double 
字符 类 型 char 
数组 
结构 体 struct 
构造 类 型 共用 体 union 











枚 举 类 型 enum 





8 oy sa es O 


指针 类 型 


空 类 型 void 


1.4. 1， 数 据 类 型 与 “模子 ” 











short, int, long, char, float, double 这 六 个 关键 字 代 表 C 语言 里 的 六 种 基本 数据 类 型 。 


怎么 去 理解 它们 呢 ? 举 个 例子 见 过 藉 煤 球 的 那个 东西 吧 ? ( 没 见 过 ? 煤 球 总 见 过 吧 )。 那 个 
东西 叫 藉 煤 器 ， 拿 着 它 在 和 好 的 煤 堆 里 这 么 一 咱 ， 一 个 煤 球 出 来 了 。 半 径 12cm, 12 个 孔 。 
不 同型 号 的 厌 煤 器 味 出 来 的 煤 球 大 小 不 一 样 ， 孔 数 也 不 一 样 。 这 个 夭 煤 器 其 实 就 是 个 模子 。 


现在 我 们 联想 一 下 ，short、int、long、char、float、double 这 六 个 东 东 是 不 是 很 像 不 同 
类 型 的 夭 煤 器 啊 ? 拿 着 它们 在 内 存 上 味 味 味 ， 不 同 大 小 的 内 存 就 分 配 好 了 ， 当 然 别 忘 了 给 
它们 取 个 好 听 的 名 字 。 在 32 位 的 系统 上 short 味 出 来 的 内 存 大 小 是 2 byte; int 味 出 来 的 
内 存 大 小 是 4 个 byte;1long 味 出 来 的 内 存 大 小 是 4 个 byte;float 味 出 来 的 内 存 大 小 是 4 个 byte; 
double 味 出 来 的 内 存 大 小 是 8 个 byte char 味 出 来 的 内 存 大 小 是 1 个 byte。 (注意 这 里 指 一 
般 情况 ， 可 能 不 同 的 平台 还 会 有 所 不 同 ， 有 具体 平台 可 以 用 sizeof 关键 字 测 试 一 下 ) 


很 简单 吧 ? 味 味 味 很 夷 吧 ? 是 很 简单 ,也 确实 很 爽 ,但 问题 就 是 你 味 出 来 这 么 多 内 存 块 ， 
你 总 不 能 给 他 取 名 字 叫 做 x1,x2,x3,x4,x5... 或 者 长 江 1 号 ,长 江 2 号 ... 吧 。 它 们 长 得 这 么 像 (不 
是 你 家 的 老大 ， 老 二 ， 老 三 …)， 过 一 阵子 你 就 会 筷 了 到 底 哪个 名 字 和 哪个 内 存 块 匹 配 了 (到 
底 谁 嫁 给 谁 了 啊 ? ^ 人 ^)。 所 以 昵 ， 给 他 们 取 一 个 好 的 名 字 绝 对 重要 。 下 面 我 们 就 来 研究 研究 
取 什 么 样 的 名 字 好 。 


















































































































































































































































1. 4. 2， 变 量 的 命名 规则 


一 般 规则 : 
【规则 1-1】 命名 应 当 直 观 且 可 以 拼 读 ， 可 望 文 知 意 ， 便 于 记忆 和 阅读 。 
标识 符 最 好 采用 英文 单词 或 其 组 合 , 不 允许 使 用 拼音 。 程序 中 的 英文 单词 一 般 不 要 太 复 
杂 ， 用 词 应 当 准 确 。 
【规则 1-2】 命 名 的 长 度 应 当 符 合 “min-length && max-information ”原则 。 


C 是 一 种 简洁 的 语言 , 命名 也 应 该 是 简洁 的 。 例 如 变量 名 MaxVal 就 比 
MaxValueUntilOverflow 好 用 。 标 识 符 的 长 度 一 般 不 要 过 长 ， 较 长 的 单词 可 通过 去 掉 “ 元 音 ” 
形成 缩写 。 
另外 ， 英 文 词 尽量 不 缩写 ， 特 别 是 非常 用 专业 名 词 ， 如 果 有 缩写 ， 在 同一 系统 中 对 同一 
单词 必须 使 用 相同 的 表示 法 ， 并 且 注 明 其 意思 。 


【规则 1-3】 当 标识 符 由 多 个 词组 成 时 ， 每 个 词 的 第 一 个 字母 大 写 ， 其 余 全 部 小 写 。 比 如 




































































































































































































































































int CurrentVal: 


这 样 的 名 字 看 起 来 比较 清晰 ， 远 比 一 长 串 字符 好 得 多 。 


















































【规则 1-4】 尽 量 避 免 名 字 中 出 现 数字 编号 ， 如 Valuel,Value2 等 ， 除 非 逻辑 上 的 确 需 要 编 
号 。 比 如 驱动 开发 时 为 管 脚 命 名 ， 非 编号 名 字 反 而 不 好 。 

初学 者 总 是 喜欢 用 带 编 号 的 变量 名 或 函数 名 , 这 样子 看 上 去 很 简单 方便 , 但 其 实 是 一 颗 
颗 定 时 炸弹 。 这 个 习惯 初学 者 一 定 要 改过 来 。 




































































【规则 1-5】 对 在 多 个 文件 之 间 共 同 使 用 的 全 局 变量 或 函数 要 加 范围 限定 符 (建议 使 用 模块 名 
(缩写 ) 作 为 范围 限定 符 )。(GUL_ , etc) 
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标识 符 的 命名 规则 : 


【规则 1-6】 标 识 符 名 分 为 两 部 分 : 规范 标识 符 前 缀 (后 缀 ) + 含义 标识 。 非 全 局 变量 可 以 
不 用 使 用 范围 限定 符 前 级 。 























过 












范围 限定 符 前 缀 
下 划 线 


ーーー ジ ーー 
模块 名 缩写 | 作用 域 前 织 | sgs2emara | [ 指针 前 织 ] SERIES 














【规则 1-7] 作用 域 前 銀 命 名 規則 。 


ee 








lw 和 ooo 


【规则 1-8】 数 据 类 型 前 级 命名 规则 。 


opp P pee fo _ 
op ee pe | 
umum 
+ P | e ese | 


| mm mm 
° | | em mm | 
し トト | ememwwms ° C| 
に に | ee mm | _ 
ww T 





umum 
5 e | ee eeewws | 
moje | ee eeewwws | À 
m [e | ee mm | 


16 fp function void(* fpGetModeFuncList_al])( void ) 
point 


当 目 定义 
结构 数据 
typedef struct SM_EventOpt 类 型 时 使 


( 


unsigned char 

















typedef 


enum/struct/u . A 
unsigned int 





nion 
char 


}SM_EventOpt_st,*SM_EventOpt_pst; 


















































【规则 1-9】 含 义 标识 命名 规则 ， 变 量 命名 使 用 名 词性 词组 ， 函 数 命名 使 用 动词 性 词组 。 
例 如 : 








目标 词 | 动词 (的 过 去 分 词 ) | 状语 “| 目的 地 
寸 

















动词 (一 般 现 时 ) 目标 词 状语 “| 目的 地 








— 
DeleteDataFromSD | Delete Data From SD J SD H 
除数 据 


函数 含义 标识 符 构 成 ; 
































动词 (一 般 现 时 )+ 目 标 词 +[ 状 语 ]+[ 目 的 地 ]; 








【规则 1-10] 程序 中 不 得 出 现 仅 靠 大 小 写 











例如 : intx, X; 变量 x 





与 X 容易 混 清 


void foo(int x); 函数 foo 与 FOO 容易 混淆 


void FOO(float x); 


这 里 还 有 一 个 要 特 


区 分 的 相似 的 标识 符 。 





别 注意 的 就 是 1 (数 字 1) 和 1 (小 写字 母 1) 之 间 ，0 (数 字 0) Mo 
































(小 写字 母 o) 之 间 的 
了 一 次 。 


【规则 1-11】 一 个 函数 名 禁 1 


例 如 : 


#include "c_standards.h 


void foo(Gntp_1) 
{ 


Intx=D_l: 


void static_p(Vod) 


( 


int foo = lu; 








1 


区 别 。 这 两 对 真是 很 难 





区 分 的 ， 我 曾经 的 一 个 同事 就 被 这 个 问题 折腾 








上 被 用 于 其 它 之 处 。 









































【规则 1-12】 所 有 宏 定 义 、 枚 举 常数 、 只 读 变 量 全 用 大 写字 母 命名 ， 用 下 划 线 分 割 单词 。 





例如 : 


const int MAX LENGTH = 100; // 这 不 是 常量 ， 而 是 一 个 只 读 变 上 





#define FILE_PATH “/usr/tmp” 


【规则 1-13】 考 虑 到 习惯 性 问题 ， 局 部 变量 




















为 循环 变量 使 用 。 

































































\ 体 请 往 后 看 





pad 









































一 定 不 要 写 出 如 下 这 样 的 代码 : 


int p; 
char i; 
int c; 


char * a; 














中 可 采用 通用 的 命名 方式 ， 仪 限于 n、i、j 等 作 

















一 般 来 说 习惯 上 用 n,m,i,j,k 等 表示 int 类 型 的 变量 ; c, ch 等 表示 字符 类 型 变量 ，a 等 
示 数 组 ; p 等 表示 指针 。 当 然 这 仅仅 是 一 般 习 惯 ， 除 了 ijk 等 可 以 用 来 表示 循环 变量 外 ， 别 
的 字符 变量 名 尽量 不 要 使 用 。 







































































【规则 1-14】 定 义 变量 的 同时 于 万 千 万 别 筷 了 初始 化 。 定 义 变量 时 编译 器 并 不 一 定 清空 了 
这 块 内 存 ， 它 的 值 可 能 是 无 效 的 数据 。 


这 个 问题 在 内 存 管理 那 章 有 非常 详细 的 讨论 ， 请 参看 。 

















































































































【规则 1-15】 不 同类 型 数据 之 间 的 运算 要 注意 精度 扩展 问题 ， 一 般 低 精 度数 据 将 向 高 精度 
数据 扩展 。 





5， 最 冤枉 的 关键 字 ----sizeof 


1. 5. 1， 常 年 被 人 误 认 为 函数 



































sizeof 是 关键 字 不 是 函数 ， 其 实 就 算 不 知道 它 是 否 为 32 个 关键 字 之 一 时 ， 我 们 也 可 以 
昔 助 编译 器 确定 它 的 身份 。 看 下 面 的 例子 : 


Int i=0; 























= 























A),sizeof(int); B), sizeof); C), sizeof int; D), sizeof i; 
FZL, 32 位 系统 下 A)，B) 的 值 为 4。 那 CO) 的 呢 ? D) 的 呢 ? 


在 32 位 系统 下 ， 通 过 Visual C++6.0 或 任意 一 编译 器 调试 ， 我 们 发 现 D) 的 结果 也 为 4。 
IË? sizeof 后 面 的 括号 呢 ? 没有 括号 居然 也 行 ， 那 想 想 ， 函 数 名 后 面 没 有 插 号 行 吗 ?由 此 轻 
易 得 出 sizeof 绝 非 函数 。 


好 ， 再 看 C)。 编 译 器 怎么 怎么 提示 出 错 呢 ? 不 是 说 sizeof 是 个 关键 字 ， 其 后 面 的 括号 
可 以 没有 么 ? 那 你 想 想 sizeof int 表示 什么 啊 ? int 前 面 加 一 个 关键 字 ? 类 型 扩展 ”明显 不 
正确 ， 我 们 可 以 在 int 前 加 unsigned, const 等 关键 字 但 不 能 加 sizeof。 好 ， 记 住 : sizeof 在 
计算 变量 所 占 空间 大 小 时 , 括 号 可以 省略 , 而 计算 类 型 (模子 ) 大 小 时 不 能 省 略 。 一 般 情況 下 , 
咱 也 别 偷 这 个 懒 ,乖乖 的 号 上 括号 ， 继 续 装 作 一 个 “函数 ” 做 一 个 “ 披 着 函数 皮 的 关键 字 ”。 
做 我 的 关键 字 ， 让 人 家 认为 是 函数 去 吧 。 












































































































































































































































1.5.2, sizeof (int) *p 表示 什么 意思 ? 














sizeof (int) *p 表示 什么 意思 ? 
留 几 个 问题 (讲解 指针 与 数组 时 会 详细 讲解 )，32 位 系统 下 : 
int *p = NULL: 


sizeof(p) 的 值 是 多 少 ? 


















































sizeof(*p) 呢 ? 


int a[100]; 
sizeof (a) 的 值 是 多 少 ? 
sizeof(a[100]) 呢 ? /请 尤其 注意 本 例 。 


sizeof(&a) 呢 ? 


























sizeof(&za[0]) 呢 ? 


intb[100]; 
Void fun(int b[100]) 
{ 





sizeof(b):// sizeof (b) 的 值 是 多 少 ? 


1. 4，signed、unsigned 关键 字 





我 们 知道 计算 机 底层 只 认识 0、1. 任 何 数 据 到 了 底层 都 会 变 计 算 转 换 成 0、1. 那 负数 怎么 
存储 呢 ? 肯定 这 个 “-” 号 是 无 法 存 入 内 存 的 ， 怎 么 办 ? 很 好 办 ， 做 个 标记 。 把 基本 数据 类 
型 的 最 高 位 腾 出 来 ， 用 来 存 符号 ， 同 时 约定 如 下 : 最 高 位 如 果 是 1， 表 明 这 个 数 是 负数 ， 其 
值 为 除 最 高 位 以 外 的 剩余 位 的 值 添 上 这 个 “-” 号 ， 如 果 最 高 位 是 0， 表 明 这 个 数 是 正 数 ， 
其 值 ， 

































































直 为 除 最 高 位 以 外 的 剩余 位 的 值 。 





























这 样 的 话 ， 一 个 32 位 的 signed int 类 型 整数 其 值 表示 法 范围 为 : -2 一 2 -1; 8 位 的 






































char 类 型 数 其 值 表示 的 范围 为 -2 ~2 -1。 一 个 32 位 的 unsigned int 类 型 整数 其 值 表示 法 

















范围 为 : 0~ 22 -1; 8 位 的 char 类 型 数 其 值 表 示 的 范围 为 0~ 2 -1。 同 样 我 们 的 signed 关 
键 字 也 很 宽 恒 大 量 ， 你 也 可 以 完全 当 它 不 存在 ， 编 译 器 缺 省 默认 情况 下 数据 为 signed 类 型 





















































上 面 的 解释 很 容易 理解 ， 下 面 就 考虑 一 下 这 个 问题 




















int main() 
{ 
char a[1000]; 
inti; 
for(i=0; 1 く 1000: i++) 
{ 
ali] = -1-i; 
) 


printf("%d",strlen(a)): 


return 0: 


) 











此 题 看 上 去 真 的 很 简单 ， 但 是 却 鲜 有 人 答对 。 答 案 是 255。 别 惊讶 ， 我 们 先 分 析 分 析 。 























for 循环 内 ， 当 i 的 值 为 0 时 ， 
我 们 知道 在 计算 机 系统 中 ， 数 值 一 得 
以 将 符号 位 和 其 它 位 统一 处 理 ; 









































シッ 


a[0] 的 值 为 -1。 关 键 就 是 -1 在 内 存 
LE 用 补 码 来 表示 (存储 )。 主 要 原 


同时 ， 减 法 也 可 按 加 法 来 处 理 。 另 外 ， 两 个 月 
































相 加 时 ， 如 果 最 高 位 ( 
补 码 : 


付 


号 位 ) 有 进位 ， 则 进位 被 舍弃 。 
符号 位 为 1， 其 余 位 为 该 数 绝对 值 的 原 码 按 位 取 反 ， 














按照 负数 补 码 的 规 由 











时 ，a[127] 的 值 为 -128， 而 -128 是 char 类 型 数据 能 对 








的 值 肯 定 不 能 是 -129。 因 
数据 只 有 8 位 ， 所 以 最 高 位 被 丢弃 。 剩 下 的 
当 i 继续 增加 到 255 的 时 候 ，-256 的 补 码 的 
码 的 低 8 位 全 为 1， 即 低 八 位 的 补 码 为 0xff, 

按照 上 面 的 分 析 ， 



































1， 可 以 知道 -1 的 补 码 为 0xff，-2 








算 字符 串 长 度 的 ， 并 不 包含 字符 串 最 后 的 “0”。 而 














而 如 何 存储 。 























因 是 使 用 补 码 ， 可 

















昌 补 码 


示 的 数 








正 数 的 补 码 与 



































8 位 是 原 





如 此 又 开始 一 轮 新 的 循环 








判断 一 个 字符 囊 





是 否 遇 到 “\0”。 如 果 遇 到 “\0”， 则 认为 本 字符 串 结束 。 











分 析 到 这 里 ， 
类 型 默认 情况 下 是 有 符号 























出 。 另 外 还 要 清楚 的 就 是 负数 的 补 码 怎 么 表示 。 FEH 





留 三 个 问题 : 














1)， 按 照 我 们 上 面 的 解释 ， 那 -0 和 +0 在 内 存 里 面 分 别 怎么 存储 ? 








2)，inti = -20: 
unsigned j=10; 


计 j 的 值 为 多 少 ? 为 什么 ? 
3), 下 面 的 代码 有 什么 问题 ? 


unsigned 1 : 
for (1=9;1>=0;i--) 
{ 























printf("%u\n",i); 
} 


1.6, if. else 组 合 























if BAIRI. R, AIRE. 
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1.6.1, bool 变量 与 “ 零 值 ”进行 比较 











bool 变量 与 “ 零 值 ”i 








行 比较 的 这 语句 怎么 写 ? 


J 
然后 整个 数 加 1。 
的 补 码 为 0xfe 
示 的 最小 的 負数 。 当 i 继续 增加 ，a[128] 
为 这 时 候 发 生 了 溢出 ，-129 需要 9 位 才能 存储 下 来 ， 而 char 类 型 
来 9 位 补 码 的 低 8 位 的 值 ， 即 0x7f。 
氏 8 位 为 0。 然 后 当 i 增加 到 256 时 ，-257 的 补 








当 i 的 值 为 127 




















a[0] 到 a[254] 里 面 的 值 都 不 为 0， 而 a[255] 的 值 为 0。strlen 函数 是 计 
是 否 结束 的 标志 就 是 看 














char 





[-128,127]， 超 出 这 个 范 
白 了 这 两 点 , 这 个 问题 


strlen(a) 的 值 为 255 应 该 完全 能 理解 了 。 这 个 问题 的 关键 就 是 要 明 
的 ， 其 表示 的 值 的 范围 ， 














围 的 值 会 产生 汶 











实 就 很 简单 了 。 








单 的 问题 : 


bool bTestFlag = FALSE;// 想 想 为 什么 一 般 初 始 化 为 FALSE 比较 好 ? 

A), if(bTestFlag == 0); if(bTestFlag == 1); 

B), if(bTestFlag == TRUE), if(bTestFlag == FLASE); 

C), if(bTestFlag); if(!bTestFlae); 
哪 一 组 或 是 那些 组 正确 呢 ? 我 们 来 分 析 分 析 : 

A) 写 法 : bTestFlag 是 什么 ? 整 型 变量 ?如 果 要 不 是 这 个 名 字 遵 照 了 前 面 的 命名 规范 ， 
肯 怕 很 容易 让 人 误会 成 整 型 变量 。 所 以 这 种 写法 不 好 。 

B) 写 法 : FLASE 的 值 大 家 都 知道 ， 在 编译 器 里 被 定义 为 0; 但 TRUE 的 值 呢 ? 都 是 1 
吗 ? 很 不 幸 ， 不 都 是 1。Visual C++ 定义 为 1， 而 它 的 同胞 兄弟 Visual Basic 就 把 TRUE 定义 
为 -1. 那 很 显然 ， 这 种 写法 也 不 好 。 

大 家 都 知道 ff 语句 是 靠 其 后 面 的 括号 里 的 表达 式 的 值 来 进行 分 支 跳 转 的 。 表 达 式 如 果 
为 真 ， 则 执行 让 语句 后 面 紧 跟 的 代码 ; 否则 不 执行 。 那 显然 ， 本 组 的 写法 很 好 ， 既 不 会 引 
起 误会 , 也 不 会 由 于 TRUE 或 FLASE 的 不 同 定义 值 而 出 错 。 记 住 : 以 后 写 代 码 就 得 这 样 写 。 

























































































































































































1.6.2, float 变量 与 “ 零 值 ”进行 比较 











float 变量 与 “ 零 值 ”进行 比较 的 让 语句 怎么 写 ? 


float fTestVal = 0.0; 








A), if(fTestVal == 0.0); if(fTestVal != 0.0); 


B), if((fTestVal >= -EPSINON) && (fTestVal <= EPSINON)): //EPSINON 为 定义 好 的 


哪 一 组 或 是 那些 组 正确 呢 ? 我 们 来 分 析 分 析 : 


float 和 double 类 型 的 数据 都 是 有 精度 限制 的 , 这 样 直 接 拿 来 与 0.0 比 , 能 正确 吗 ? 明显 
不能 , 看 例 子 : x 的 值 四 舍 五 入 精确 到 小 数 点 后 10 位 为 :3.1415926536， 你 拿 它 减 去 
0.00000000001 然后 再 四 舍 五 入 得 到 的 结果 是 多 少 ? 你 能 说 前 后 两 个 值 一 样 吗 ? 


EPSINON 为 定义 好 的 精度 ， 如 果 一 个 数落 在 [0.0-EPSINON,0.0+EPSINON] 这 个 闭 区 间 
内 ， 我 们 认为 在 某 个 精度 内 它 的 值 与 零 值 相等 ， 否 则 不 相等 。 扩 展 一 下 ， 把 0.0 替换 为 你 想 
比较 的 任何 一 个 浮 点 数 ， 那 我 们 就 可 以 比较 任意 两 个 浮 点 数 的 大 小 了 ， 当 然 是 在 某 个 精度 
内 。 

同样 的 也 不 要 在 很 大 的 浮 点 数 和 很 小 的 浮 点 数 之 间 进 行 运算 ， 比 如 : 

10000000000.00 + 0.00000000001 

这 样 计 算 后 的 结果 可 能 会 让 你 大 吃 一 惊 。 






























































































































































1. 6. 3， 指 针 变 量 与 “ 零 值 ”进行 比较 


指针 变量 与 “ 零 值 ”进行 比较 的 站 语句 怎么 写 ? 





“ 








int * p =NULL;// 定 义 指 针 一 定 要 同时 初始 化 ， 指 针 与 数组 那 章 会 详细 讲解 。 

A), if(p == 0); if(p != 0); 

B), if(p); if(!p); 

C) ,ifNULL == p); if(NULL != p); 

哪 一 组 或 是 那些 组 正确 呢 ? 我 们 来 分 析 分 析 : 

A) 写 法 : p 是 整 型 变量 ? 容易 引起 误会 , 不 好 。 尽管 NULL 的 值 和 0 一样 , 但 意义 不 同 。 
B) 写 法 : p 是 bool 型 变量 ? 容易 引起 误会 ， 不 好 。 


OSA: 这 个 写法 才 是 正确 的 ， 但 样子 比较 古怪 。 为 什么 要 这 么 写 呢 ? 是 怕 漏 写 一 个 
”号 :fp =NULD)， 这 个 表达 式 编译 器 当然 会 认为 是 正确 的 ， 但 却 不 是 你 要 表达 的 意思 。 



































































































































所 以 ， 非 常 推荐 这 种 写法 。 


1.6.4, else 到 底 与 哪个 if 配对 呢 ? 





else 常常 与 让 语句 配对 ， 但 要 注意 书写 规范 ， 看 下 面 例子 : 
if (0==x) 








if (0==y) error (); 
else{ 
//program code 
} 
这 个 else 到 底 与 谁 匹配 呢 ? 让 人 迷糊 , 尤其 是 初学 者 。 还 好 , C 语言 有 这 样 的 规定 : else 























始终 与 同一 括号 内 最 近 的 未 匹配 的 让 语句 结合 。 虽 然 老手 可 以 区 分 出 来 ， 但 这 样 的 代码 谁 
都 会 头疼 的 ， 任 何 时 候 都 别 偷 这 种 懒 。 关 于 程序 中 的 分 界 符 “{” 和 “}”， 建 议 如 下 : 


【建议 1-16】 程序 中 的 分 界 符 “{” 和 “}” 对 齐 风 格 如 下 : 


注意 下 表 中 代码 的 缩 进 一 般 为 4 个 字符 , 但 不 要 使 用 Tab 键 ， 因 为 不 同 的 编辑 器 Tab 键 定义 
的 空格 数量 不 一 样 ， 别 的 编辑 器 打开 Tab 键 缩 进 的 代码 可 能 会 一 片 混乱 。 


提 1 









































































































































昌 的 的 风格 不 提倡 的 风格 


Void Function(int x) void Function(int x){ 


{ 


/program code 


/program code } 


if (condition) if (condition){ 


{ 





//program code 


//program code }else{ 


/program code 


} 
或 : 
/program code if (condition) 

//program code 

else 

//program code 

或 : 

if (width < height) dosomething(); 
for (initialization; condition; update) for (initialization;condition; update){ 
{ /program code 


/program code } 


while (condition) while (condition){ 
{ //program code 


//program code } 


dof 


/program code 


/program code }while (condition); 


) 


while (condition): 





1.6.5, if 语句 后 面 的 分 号 
































关于 if-else 语句 还 有 一 个 容易 出 错 的 地 方 就 是 与 空 语句 的 连用 。 看 下 面 的 例子 : 


IE(NULL !=p) ; 

fun(): 
这 里 的 fanO 函 数 并 不 是 在 NULL =p 的 时 候 被 调用 ， 而 是 任何 时 候 都 会 被 调用 。 问 题 就 出 
在 证 语句 后 面 的 分 号 上 。 在 C 语言 中 ， 分 号 预示 着 一 条 语句 的 结尾 ， 但 是 并 不 是 每 条 C 语 
言语 名 都 需要 分 号 作为 结束 标志 。 让 语句 的 后 面 并 不 需要 分 号 , 但 如 果 你 不 小 心 写 了 个 分 号 ， 
编译 器 并 不 会 提示 出 错 。 因 为 编译 器 会 把 这 个 分 号 解析 成 一 条 空 语句 。 也 就 是 上 面 的 代码 实 
际 等 效 于 : 

if(NULL != p) 

{ 























































































































} 

fun(): 
这 是 初学 者 很 容易 犯 的 错误 ， 往 往 不 小 心 多 写 了 个 分 号 ， 导 致 结果 与 预想 的 相差 很 远 。 所 
以 建议 在 真正 需要 用 空 语句 时 写成 这 样 : 


























NULL: 














而 不 是 单 用 一 个 分 号 。 这 就 好 比 汇编 语言 里 面 的 空 指令 ， 比 如 ARM 指令 中 的 NOP 指令 。 
这 样 做 可 以 明显 的 区 分 真正 必须 的 空 语句 和 不 小 心 多 写 的 分 号 。 


















































1.6.6, 使用 if 语句 的 其 他 注意 事项 





【规则 1-17】 先 处 理 正常 情况 ， 再 处 理 异常 情况 。 

在 编写 代码 是 ， 要 使 得 正常 情况 的 执行 代码 清晰 ， 确 认 那 些 不 常 发 生 的 异常 情况 处 理 
代码 不 会 遮掩 正常 的 执行 路 径 。 这 样 对 于 代码 的 可 读 性 和 性 能 都 很 重要 。 因 为 ， 讶 语句 总 是 
需要 做 判断 ， 而 正常 情况 一 般 比 异常 情况 发 生 的 概率 更 大 〈 和 否则 就 应 该 把 异常 正常 调 过 来 
了 )， 如 果 把 执行 概率 更 大 的 代码 放 到 后 面 ， 也 就 意味 着 让 语句 将 进行 多 次 无 谓 的 比较 。 另 
外 ， 非 常 重 要 的 一 点 是 ， 把 正常 情况 的 处 理 放 在 这 后 面 ， 而 不 要 放 在 else 后 面 。 当 然 这 也 
符合 把 正常 情况 的 处 理 放 在 前 面 的 要 求 。 

【规则 1-18】 确 保证 和 else 子 句 没 有 和 弄 反 。 

这 一 点 初学 者 也 容易 弄 错 ， 往 往 把 本 应 该 放 在 站 语句 后 面 的 代码 和 本 应 该 放 在 else 语 
句 后 面 的 代码 弄 反 了 。 

























































































































































































1.7, switch、case 组 合 


既然 有 了 if. else 组 合 为 什么 还 需要 switch, case 组 合 呢 ? 





1. 7. 1， 不 要 拿 青 龙 优 月 刀 去 削 苹 果 





那 你 既然 有 了 菜刀 为 什么 还 需要 水 果 刀 呢 ? RAPERE KREEK AZA 
HD 去 削 苹 果 吧 。 如 果 你 真能 做 到 ， 关 二 和 爷 也 会 修 服 你 的 。^_^。 

if. else 一 般 表示 两 个 分 支 或 是 租 套 表示 少量 的 分 支 ， 但 如 果 分 支 很 多 的 话 .….…. 还 是 用 
switch、case 组 合 吧 。 其 基本 格式 为 : 












































switch(variable) 
{ 
case Valuel: 


//program code 


break: 

case Value2: 
//program code 
break: 

case Value3: 
//program code 


break: 


default: 
break: 


ーー 











很 简单 ， 但 有 两 个 规则 : 

【规则 1-19] 每 个 case 语句 的 结尾 绝对 不 要 忘 了 加 break， 否 则 将 导致 多 个 分 支 重 登 《除非 
有 意 使 多 个 分 支 重 对 )。 
【规则 1-20】 最 后 必须 使 用 default 分 支 。 即 使 程序 真 的 不 需要 default 处 理 ， 也 应 该 保留 
语句 : 


default : 















































break: 
这 样 做 并 非 了 画蛇添足， 可 以 避免 让 人 误 以 为 你 筷 了 default 处 理 。 
































1.7.2, case 关键 字 后 面 的 值 有 什么 要 求 吗 ? 











= 


























好 ， 再 问 问 : 真 的 就 这 么 简单 吗 ? 看 看 下 面 的 问题 : 

Valuel 的 值 为 0.1 行 吗 ?-0.1 Da? -1 WE? 0.1+0.9 WE? 1+2 Ba? 3/2 WE? “A ME? “A” 
呢 ? 变量 1i〈 假 设 i 已 经 被 初始 化 ) BZ? NULL WE? 等 等 。 这 些 情形 希望 你 亲自 上 机 调试 一 
下 ， 看 看 到 底 哪些 行 ， 哪 些 不 行 。 

WÈ: case 后 面 只 能 是 整 型 或 字符 型 的 常量 或 常量 表达 式 〈 想 想 字符 型 数据 在 内 存 里 
是 怎么 存 的 )。 













































































1.7.3，case 语句 的 排列 顺序 























似乎 从 来 没有 人 考虑 过 这 个 问题 ， 也 有 很 多 人 认为 case 语句 的 顺序 无 所 谓 。 但 事实 却 
不 是 如 此 。 如 果 case 语句 很 少 ， 你 也 许可 以 忽略 这 点 ， 但 是 如 果 case 语句 非常 多 ， 那 就 不 
得 不 好 好 考虑 这 个 问题 了 。 比 如 你 写 的 是 某 个 驱动 程序 ， 也 许 会 经 常 遇 到 几 十 个 case 语句 
的 情况 。 一 般 来 说 ， 我 们 可 以 遵循 下 面 的 规则 : 




























































































【规则 1-21】 按 字母 或 数字 顺序 排列 各 条 case 语句 。 
如 果 所 有 的 case 语句 没有 明显 的 重要 性 差别 ， 那 就 按 A-B-C 或 1-2-3 等 顺序 排列 case 
语句 。 这 样 做 的 话 ， 你 可 以 很 容易 的 找到 茶 条 case 语句 。 比 如 : 


switch(variable) 


( 























case A: 
//program code 
break; 

case B: 
//program code 
break; 

case C: 
//program code 


break; 


default: 


break; 


) 

【规则 1-22】 把 正常 情况 放 在 前 面 ， 而 把 异常 情况 放 在 后 面 。 

如 果 有 多 个 正常 情况 和 异常 情况 ， 把 正常 情况 放 在 前 面 ， 并 做 好 注释 ， 把 异常 情况 放 在 
后 面 ， 同 样 要 做 注释 。 比 如 : 


Switch(Variable) 


( 















































JII 


/正常 情况 开始 





case A: 
//program code 
break; 

case B: 
//program code 
break; 


/正常 情况 结束 
HH 





/异常 情况 开始 
case -1: 
//program code 
break: 


/异常 情况 结束 
HH 





default: 
break: 


【规则 1-23 】 按 执行 频率 排列 case 语句 

把 最 常 执行 的 情况 放 在 前 面 ， 而 把 最 不 常 执 行 的 情况 放 在 后 面 。 最 常 执行 的 代码 可 能 
也 是 调试 的 时 候 要 单 步 执 行 的 最 多 的 代码 。 如 果 放 在 后 面 的 话 ， 找 起 来 可 能 会 比较 困难 ， 而 
放 在 前 面 的 话 ， 可 以 很 快 的 找到 。 
























































1. 7.4， 使 用 case 语句 的 其 他 注意 事项 











【规则 1-24】 简 化 每 种 情况 对 应 的 操作 。 

使 得 与 每 种 情况 相关 的 代码 尽 可 能 的 精炼 。case 语句 后 面 的 代码 越 精炼 ，case 语句 的 结 
果 就 会 越 清 晰 。 你 想 想 ， 如 果 case 语句 后 面 的 代码 整个 屏幕 都 放 不 下 ， E IE 
看 得 很 清晰 吧 。 如 果 某 个 case 语句 确实 需要 这 么 多 的 代码 来 执行 某 个 操作 ， 那 可 以 把 这 
操作 写成 一 个 或 几 个 子 程序 ， 然 后 在 case 语句 后 面 调 用 这 些 子 程序 就 ok 了 。 一 般 来 说 cast case 
语句 后 面 的 代码 尽量 不 要 超过 20 行 。 


























































































































【规则 1-25】 不 要 为 了 使 用 case 语句 而 刻意 制造 一 个 变量 。 

case 语句 应 该 用 于 处 理 简 单 的 ， 容 易 分 类 的 数据 。 如 果 你 的 数据 并 不 简单 ， 那 可 能 使 用 让 
else if 的 组 合 更 好 一 些 。 为 了 使 用 case 而 刻意 构造 出 来 的 变量 很 容易 把 人 搞 糊涂 ， 应 该 避免 
这 种 变量 。 比 如 : 


char action = a[0]; 













































































switch (action) 
{ 
case ‘c’: 
fun1 (); 
break: 


case ‘d’: 


break: 
default: 


break: 


} 

这 里 控制 case 语句 的 变量 是 action。 而 action 的 值 是 取 字 符 数 组 a 的 一 个 字符 。 但 是 这 
种 方式 可 能 带 来 一 些 隐 含 的 错误 。 一 般 而 言 ， 当 你 为 了 使 用 case 语句 而 刻意 去 造 出 一 个 变 
量 时 ， 真 正 的 数据 可 能 不 会 按照 你 所 希望 的 方式 映射 到 case 语句 里 。 在 这 个 例子 中 ， 如 果 
用 户 输 入 字符 数组 a 里 面 存 的 是 “const” 这 个 字符 串 ， 那 么 case 语句 会 匹配 到 第 一 个 case 
上 ， 并 调用 fun1 O 函数 。 然 而 如 果 这 个 数组 里 存 的 是 别 的 以 字符 c 开头 的 任何 字符 串 〈 比 
HH: “col” “can”), case 分 支 同 样 会 匹配 到 第 一 个 case 上 。 但 是 这 也 许 并 不 是 你 想 要 的 结 
果 ， 这 个 隐 舍 的 错误 往往 使 人 抓 狂 。 如 果 这 样 的 话 还 不 如 使 用 if-else f 组 合 。 比 如 : 


if (0 == strcmp(“const”, a)) 














































































































{ 
fun10(): 
) 
else if 
{ 
) 














【规则 1-26】 把 default 子 句 只 用 于 检查 真正 的 默认 情况 。 

有 时 候 ， 你 只 剩 下 了 最 后 一 种 情况 需要 处 理 ， 于 是 就 决定 把 这 种 情况 用 default 子 句 来 
处 理 。 这 样 也 许 会 让 你 偷懒 少 敲 儿 个 字符 ， 但 是 这 却 很 不 明智 。 这 样 将 失去 case 语句 的 标 
号 所 提供 的 自 说 明 功 能 ， 而 且 也 丧失 了 使 用 default 子 句 处 理 错误 情况 的 能 力 。 所 以 ， 奉 劝 
你 不 要 偷懒 ， 老 老实 实 的 把 每 一 种 情况 都 用 case 语句 来 完成 ， 而 把 真正 的 默认 情况 的 处 理 
交 给 default 子 句 。 
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1.8, do. while. for 关键 字 








C 语言 中 循环 语句 有 三 种 : while 循环 、do-while 循环 、for 循环 。 


while 循环 : 先 判 断 while 后 面 括号 里 的 值 ， 如 果 为 真 则 执行 其 后 面 的 代码 ;否则 不 执 
行 。while (1) 表示 死 循环 。 死 循环 有 没有 用 呢 ? 看 下 面 例子 : 


比如 你 开发 一 个 系统 要 日 夜 不 停 的 运行 ， 但 是 只 有 操作 员 输 入 某 个 特定 的 字符 “#” 才 
可 以 停 下 来 。 


while (1) 
{ 







































































if(‘#’== GetInputChar()) 
{ 
break: 


1.8.1, break 与 continue 的 区 別 








break 关键 字 很 重要 ， 表 示 终 止 本 层 循环 。 现 在 这 个 例子 只 有 一 层 循 环 ， 当 代码 执行 到 
break 时 ， 循 环 便 终 止 。 


如 果 把 break 换 成 continue 会 是 什么 样子 呢 ? continue 表示 终止 本 次 (本 轮 ) 循环 。 当 
代码 执行 到 continue 时 ， 本 轮 循环 终止 ， 进 入 下 一 轮 循环 。 

while (1) 也 有 写成 while(true) 或 者 while(1==1) 或 者 while((booD 1) 等 形式 的 , 数 果 一 
样 。 

do-while 循环 ， 先 执行 do 后 面 的 代码 ， 然 后 再 判断 while 后 面 括号 里 的 值 ， 如 果 为 真 ， 
循环 开始 ; 和 否则， 循环 不 开始 。 其 用 法 与 while 循环 没有 区 别 ， 但 相对 较 少 用 。 

for 循环 : for 循环 可 以 很 容易 的 控制 循环 次 数 ， 多 用 于 事先 知道 循环 次 数 的 情况 下 。 


留 一 个 问题 : 在 switch case 语句 中 能 否 使 用 continue 关键 字 ? 为 什么 ? 














































































































1. 8. 2， 循 环 语 句 的 注意 点 























【建议 1-27】 在 多 重 循环 中 ， 如 果 有 可 能 ， 应 当 将 最 长 的 循环 放 在 最 内 层 ， 最 短 的 循环 放 
在 最 外 层 ， 以 减少 CPU 跨 切 循环 层 的 次 数 。 











例如 : 


长 循环 在 最 内 层 ， 效 率 高 长 循环 在 最 外 层 ， 效 率 低 


for (col=0: col<5; col++ ) for (row=0; row<100; row++) 


{ ( 


for (row=0: row く 100: row++) for ( col=0: col<5; col++ ) 


{ { 


sum = sum + alrowllcoll: sum = sum + a[row][col]; 

















【建议 1-28】 建议 for 语句 的 循环 控制 变量 的 取 值 采用 “ 半 开 半 闭 区 间 ” 写 法 。 


半 开 半 闭 区 间 写 法 和 闭 区 间 写 法 虽然 功能 是 相同 , 但 相 比 之 下 , 半 开 半 闭 区 间 写 法 写法 更 加 
直观 。 


半 开 半 闭 区 间 写 法 闭 区 间 写 法 






































for (n = 0; n < 10; n++) for (n = 0; n <= 9; n++) 








【规则 1-29】 不能 在 for 循环 体内 修改 循环 变量 ， 防 止 循环 失控 。 
for (n = 0; n < 10; n++) 


( 








n = 8:// 不 可 ， 很 可 能 违 























【规则 1-30】 循 环 要 尽 可 能 的 短 ， 要 使 代码 清晰 ， 一 目 了 然 。 






































如 果 你 写 的 一 个 循环 的 代码 超过 一 显示 屏 ， 那 会 让 读 代 码 的 人 发 狂 的 。 解 决 的 办 法 由 
两 个 : 第 一 ， 重 新 设计 这 个 循环 ， 确 认 是 否 这 些 操作 都 必须 放 在 这 个 循环 里 ， 第 二 ， 将 这 些 
代码 改写 成 一 个 子 函数 , 循环 中 只 调用 这 个 子 函 数 即 可 。 一 般 来 说 循环 内 的 代码 不 要 超过 20 
行 。 

【规则 1-31】 把 循环 嵌 套 控制 在 3 层 以 内 。 

国外 有 研究 数据 表明 ， 当 循环 侍 套 超过 3 层 ， 程 序 员 对 循环 的 理解 能 力 会 极 大 的 降低 。 
如 果 你 的 循环 峙 套 超过 3 层 ， 建 议 你 重新 设计 循环 或 是 将 循环 内 的 代码 改写 成 一 个 字 函 数 。 





































































































1. 9，goto 关键 字 


一 般 来 说 ， 编 码 的 水 平 与 goto 语句 使 用 的 次 数 成 反比 。 有 的 人 主张 慎 用 但 不 禁用 goto 
语句 ， 但 我 主张 禁用 。 关 于 goto 语句 的 更 多 讨论 可 以 参看 Steve McConnell 的 名 著 《Code 
Complete. Second Edition}. 









































【规则 1-32】 禁 用 goto 语句 。 

自从 提倡 结构 化 设计 以 来 ，goto 就 成 了 有 争议 的 语句 。 首 先 ， 由 于 goto 语句 可 以 灵活 
跳 转 ， 如 果 不 加 限制 ， 它 的 确 会 破坏 结构 化 设计 风格 ; 其 次 ，goto 语句 经 常 带 来 错误 或 隐 
患 。 它 可 能 跳 过 了 变量 的 初始 化 、 重 要 的 计算 等 语句 ， 例 如 : 



























































struct student *p = NULL; 


goto state; 





p = (struct student *)malloc(...); /被 goto 跳 过 ,没有 初始 化 


State: 


使用 p 指向 的 内 存 里 的 值 的 代码 





/ 


~ 

















如 果 编 译 器 不 能 发 觉 此 类 错误 ， 每 用 一 次 goto 语句 都 可 能 留 下 隐患 。 











1. 10，void 关键 字 

















void 有 什么 好 讲 的 呢 ? 如 果 你 认为 没有 ， 那 就 没有 ; 但 如 果 你 认为 有 ， 那 就 真 的 有 。 
有 点 像 “ 色 即 是 空 ， 空 即 是 色 ”。 






































1.10.1, void a? 











void 的 字面 意思 是 “ 空 类 型 ”, void * 则 为 “ 空 类 型 指针 ”, void * 可 以 指向 任何 类 型 的 数据 。 

void 几乎 只 有 “注释 "和 限制 程序 的 作用 ， 因 为 从 来 没有 人 会 定义 一 个 void $E, 看 看 下田 
的 例子 : 

Void a: 

Visual C++6.0 上 ， 这 行 语句 编译 时 会 出 错 ， 提 示 “illegal use of type "void”。 不 过 ， 即 使 
void a 的 编译 不 会 出 错 ， 它 也 没有 任何 实际 意义 。 

void 真正 发 挥 的 作用 在 于 : 

(1) 对 函数 返回 的 限定 ; 

(2) 对 函数 参数 的 限定 。 

众所周知 ,如 果 指 针 pl 和 p2 的 类 型 相同 , 那么 我 们 可 以 直接 在 pl 和 p2 间 互 相 赋值 ; 
如果 pl 和 p2 指向 不 同 的 数据 类 型 ， 则 必须 使 用 强制 类 型 转换 运算 符 把 赋值 运算 符 右边 的 
指针 类 型 转换 为 左边 指针 的 类 型 。 


例如 : 
float *pl; 


















































































































































int *p2; 
pl =p2: 








其 中 pl = p2 语句 会 编译 出 错 ， 提 示 “=' : cannot convert from 'int *' to float*”， 必 须 改 为 : 
p1 = (float *)p2: 

而 void * 则 不 同 ， 任 何 类 型 的 指针 都 可 以 直接 赋值 给 它 ， 无 需 进 行 强制 类 型 转换 ; 
void *pl; 




















int *p2; 
pl =p2: 
但 这 并 不 意味 着 ，void * 也 可 以 无 需 强 制 类 型 转换 地 赋 给 其 它 类 型 的 指针 。 因 为 “ 空 类 型 "可 
以 包容 “有 类 型 ”而 “有 类 型 *” 则 不 能 包容 “ 空 类 型 >。 比如, 我 们 可 以 说 “男人 和 女人 都 是 人 ”， 
但 不 能 说 “人 是 男人 ?或 者 "人 是 女人 ”。 下 面 的 语句 编译 出 错 : 
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void *pl; 
int *p2; 
p2= pl; 
提示 ”=' : cannot convert from "void *' to 'int *'”, 


1.10.2, void 修饰 函数 返回 值 和 参数 


【规则 1-33】 如 果 函 数 没 有 返回 值 ， 那 么 应 声明 为 void 类 型 















































在 C 语言 中 ， 几 不 加 返回 值 类 型 限定 的 函数 ， 就 会 被 编译 占 作 为 返回 整 型 值 处 理 。1 



























































许多 程序 员 却 误 以 为 其 为 void 类 型 。 例 如 : 
add (inta,intb ) 
{ 





部 








return a+ b; 


int main(int argc, char* argv[]) /甚至 很 多 人 以 为 main 函数 无 返回 值 
/或 是 为 void 型 的 








printf ( "2 + 3 = %d", add ( 2, 3) ): 
) 
星 序 运 行 的 结果 为 输出 : 2+3=5 
这 说 明 不 加 返回 值 说 明 的 函数 的 确 为 int 函数 。 
因此 ， 为 了 避免 混乱 ， 我 们 在 编写 C 程序 时 ， 对 于 任何 函数 都 必须 一 个 不 漏 地 指定 大 


` 





SB 



























































类 型 。 如 果 函 数 没 有 返回 值 ， 一 定 要 声明 为 void 类 型 。 这 既是 程序 良好 可 读 性 的 需要 ， 也 
编程 规范 性 的 要 求 。 另 外 ， 加 上 void 类 型 声明 后 ， 也 可 以 发 挥 代码 的 “ 自 注 释 ”" 作 用 。 所 
谓 的 代码 的 “ 自 注 释 ” 即 代码 能 自己 注释 自己 。 
【规则 1-34】 如 果 函 数 无 参数 ,那么 应 声明 其 参数 为 void 
在 C++ 语言 中 声明 一 个 这 样 的 函数 : 
int function(Void) 
( 
return 1: 
} 
则 进行 下 面 的 调用 是 不 合法 的 : function(2); 
因为 在 C++ 中 ， 函 数 参数 为 void 的 意思 是 这 个 函数 不 接受 任何 参数 。 
但 是 在 Turbo C 2.0 中 编译 : 
#include "stdio.h" 
fun() 


( 
































am 



















































































return 1; 


) 


main() 


( 
printf("%d",fun(2)); 
getchar(): 


) 











编译 正确 且 输 出 1， 这 说 明 ， 在 C 语言 中 ， 可 以 给 无 参数 的 函数 传送 任意 类 型 的 参数 ， 
























































1.10.3, void 指针 











站 ， 若 函数 不 接受 个 








【规则 1-35】 千 万 小 心 又 小 心 使 用 void 指针 类 型 。 
按照 ANSI(American National Standards Institute) 标 准 ， 不 能 对 void 指针 进行 算法 操作 ， 





即 下 列 操作 都 是 不 合法 的 


void * pvoid; 


pvoid++; //ANSI: 错误 


pvoid += 1; /ANSI: 错误 
ANSI 标准 之 所 以 这 样 认定 ， 是 因为 它 坚 持 : 进行 算法 操作 的 指针 必须 是 确定 知道 其 指 








向 数据 类 型 大 小 的 。 也 就 是 说 必 须知 道内 存 目的 地 ] 


例如 : 
int *pint; 
pint++; /ANSI: 正确 






































操作 与 char * 一 致 。 因 此 








pvoid++; //GNU: 正确 
pvoid += 1; /GNU: 正确 
在 实际 的 程序 设计 中 ， 为 符合 ANSI 标准 ， 并 提高 程序 的 可 移植 性 ， 我 们 可 以 这 样 编写 











实现 同样 功能 的 代码 : 


void * pvoid; 








人 









































(char *)pvoid++; //ANSI: 正确 ; GNU: 1 
(char *)pvoid += 1; /ANSI: 错误 ; GNU: 正确 


GNU 和 ANSI 还 有 一 些 


























但 是 大 名 易 易 的 GNU(GNU'S Not Unix 的 递归 缩写 ) 则 不 这 么 认定 , 它 指定 void * 的 算法 
下 列 语句 在 GNU 编译 器 中 皆 正 确 : 




















E 确 

















址 的 确切 值 。 


但 是 在 C++ 编译 器 中 编译 同样 的 代码 则 会 出 错 。 在 C++ 中 ， 不 能 向 无 参数 的 函数 传送 任何 
参数 ， 出 错 提示 “fun' : function does not take 1 parameters”, 


所 以 ， 无 论 在 C 还 是 C++ F 何 参数 ， 一 定 要 指明 参数 为 void。 



















































































区 别 ， 总 体 而 言 ，GNU $X ANSI 更 “开放 ”*， 提 供 了 对 更 多 语法 
的 支持 。 但 是 我 们 在 真实 设计 时 ， 还 是 应 该 尽 可 能 地 符合 





ANSI 标准 。 





【规则 1-36】 如 果 函 数 的 参数 可 以 是 任意 类 型 指针 ， 那 么 应 声明 其 参数 为 void *。 
典型 的 如 内 存 操作 函数 memcpy 和 memset 的 函数 原型 分 别 为 : 


void * memcpy(void *dest, const void *src, size_t len); 


void * memset ( void * buffer, int c, size_t num ): 


这 样 ， 任 何 类 型 的 指针 都 可 以 传 入 memcpy 和 memset 中 ， 这 也 真实 地 体现 了 内 存 操作 






































int IntArray_a[100]; 

















函数 的 意义 , 因为 它 操 作 的 对 象 仅 仅 是 一 片 内 存 , 而 不 论 这 片 内 存 是 什么 类 型 。 如 果 memcpy 
和 memset 的 参数 类 型 不 是 void *, 而 是 char *, 那 才 叫 真 的 奇怪 了 ! 这 样 的 memcpy 和 memset 
明显 不 是 一 个 “纯粹 的 ， 脱 离 低级 趣味 的 ”函数 ! 
下 面 的 代码 执行 正确 : 


列子 : memset 接受 任意 类 型 指针 


memset (IntArray_a, 0, 100*sizeof(int) ): /将 IntArray_a 清 0 





例 子 : memcpy 接受 任意 类 型 指针 


int destIntArray_a[100], srcintarray_a[100]: 





/将 srcintarray_a 拷贝 给 destIntArray_a 

memcpy (destIntArray_a, Srcintarray_a, 100*sizeof(int) ); 
有 趣 的 是 ，memcpy 和 memset 函数 返回 的 也 是 void * 类 型 ， 标 准 库 函 数 的 编写 者 都 不 是 一 
般 人 。 


1. 10. 4，void 不 能 代表 一 个 真实 的 变量 

【规则 1-37】 void 不 能 代表 一 个 真实 的 变量 。 
因为 定义 变量 时 必须 分 配 内 存 空间 ， 定 义 void 类 型 变量 ， 编 译 器 到 底 分 配 多 大 的 内 存 呢 。 
下 面 代码 都 企图 让 void 代表 一 个 真实 的 变量 ， 因 此 都 是 错误 的 代码 : 

void a: /错误 

function(void a); /错误 

void 体现 了 一 种 抽象 , 这 个 世界 上 的 变量 都 是 “有 类 型 ”的 ,譬如 一 个 人 不 是 男人 就 是 女 
人 (人 妖 不 算 )。 

void 的 出 现 只 是 为 了 一 种 抽象 的 需要 , 如 果 你 正确 地 理解 了 面向 对 象 中 “抽象 基 类 ”的 概 
念 ， 也 很 容易 理解 void 数据 类 型 。 正 如 不 能 给 抽象 基 类 定义 一 个 实例 ， 我 们 也 不 能 定义 一 
个 void《〈 让 我 们 类 比 的 称 void 为 “抽象 数据 类 型 ") 変量 。 

void 简单 吧 ? 到 底 是 “ 色 ” 还 是 “ 空 ” 呢 ? 
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1.10, return 关键 字 





return 用 来 终止 一 个 函数 并 返回 其 后 面 跟着 的 值 。 
return (Val); // 此 括号 可 以 省 略 。 但 一 般 不 省 略 ， 尤 其 在 返回 一 个 表达 式 的 值 时 。 
return 可 以 返回 些 什么 东西 呢 ? 看 下 面 例 子 : 












































char * Func(void) 


{ 
char str[30]; 


return str; 
) 
str 属于 局 部 变量 ， 位 于 栈 内 存 中 ， 在 Func 结束 的 时 候 被 释放 ， 所 以 返回 str 将 导致 错误 。 
各 向 “ 栈 内 存 ” 的 “指针 ” 因为 该 内 存在 函数 体 结束 时 
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【规则 1-38】 return 造 句 不可 返 操 
被 自动 销毁 。 


留 个 问题 : 








return ; 


这 个 语句 有 问题 吗 ? 如 果 没 有 问题 ， 那 返回 的 是 什么 ? 









































1.11, const 关键 字 也 许 该 被 替换 为 reado1ny 








const 是 constant 的 缩写 ， 是 恒定 不 变 的 意思 ， 也 翻译 为 常量 、 和 常数 等 。 
因为 这 一 点 ， 很 多 人 都 认为 被 const 修饰 的 1 
g 被 使用 , 








E, 其 


ター プ で 


的 变 





值 在 


i 译 时 






































这 个 关键 字 应 该 被 蔡 换 为 readonly。 那 么 这 个 关键 字 有 什么 














不 外 
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JE r Æo 














这 是 不 精确 





很 不 幸 ， 正 是 
的 ， 精 确 的 说 应 该 是 只 读 



































因为 编译 器 在 编译 时 不 入 





1 道 其 存储 的 内 容 。 或 许 当 初 





























const 推出 的 初始 目的 ， 
我 们 看 看 它 与 define 宏 的 
章 前 面 看 


















































1. 


区 





正 是 为 了 取代 预 s 
别 。 





i 译 指令 ， 




















11.1, const 修饰 的 只 读 变 量 














定义 const 只 读 变 量 ， 具 有 不 可 变性 。 


例如 : 


const int Max=100; 


intArray[Max]; 





这 里 请 在 Visual C++6.0 里 分 别 

















Æ 
读 属性 罢了 ， 而 在 C++ 里 ， 


` 














注意 : const 修饰 的 只 读 变 量 必须 在 定义 的 同时 初始 化 ， 
























































处 和 意义 呢 ? 














消除 它 的 人 





(很 多 人 误 以 为 define 是 关键 字 ， 在 这 里 我 提醒 你 
看 32 个 关键 字 里 是 否 有 define), 


创建 .c 文件 和 .cpp 文件 测试 一 
编译 器 会 提示 出 错 , 而 在 .cpp 文件 中 则 顺利 
其 元 素 的 个 数 。 这 也 从 侧面 证 实在 C 语言 中 ，const 1 








决 点 ， 同 时 继承 它 的 优点 。 
于 回 到 本 


























下 。 你 会 发 现在 .c 文件 中 ， 



































运行 。 为 什么 呢 ? 我 们 知道 定义 一 个 数组 必须 指 
BIRI Max 仍然 是 变量 ， 只 不 过 是 只 
扩展 了 const 的 含义 ， 这 里 就 不 讨论 了 。 
想 想 为 什么 ? 
留 一 个 问题 ，case 语句 后 面 是 否 可 以 是 const 修饰 的 只 读 变量 呢 ? 请 动手 测试 一 下 。 
1. 11. 2， 节 省 空间 ， 避 免 不 必 要 的 内 存 分 配 ， 同 时 提高 效率 
编译 器 通常 不 为 普通 const 只 读 变 量 分 配 存储 空间 ， 而 是 将 它们 保存 在 符号 表 中 ， 这 使 



































MA 昌 デラ 








FA 
例 如 : 
#define M 3 // 宏 常量 
constint N=5; /此 时 并 未 将 N 放 入 内 存 中 



















































































种 分 配 ! 


int i=N /此 时 为 N 分 配 内 存 , 以 后 不 # 

int Il=M // 预 编译 期 间 进行 宏 替 换 ， 分 配 内 存 
intj= 7 没有 内 存 分 配 

int JEM /再 进行 宏 蔡 换 ， 又 一 次 分 配 内 存 ! 























const 定义 的 只 读 变 量 从 》 

















的 角度 来 看 , 只 





C 




















一 样 给 出 的 是 立即 数 ， 所 以 ，const 定义 的 只 读 变 量 厂 
存放 在 
译 阶 段 进 行 蔡 换 ， 而 const 修饰 的 
而 const 修饰 的 只 读 变量 














E, 





它 是 全 局 的 只 读 变 
#define Z Æ EMI 


#define 宏 没 有 类 型 ， 
































成 为 一 个 编译 期 间 的 值 ， 没 有 了 存储 与 读 内 存 的 操作 ， 使 得 它 的 效率 也 很 高 。 
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EE; 
























































是 给 出 了 对 应 的 内 存 地 址 , 而 不 是 象 #define 
E 程 序 运行 过 程 中 
态 区 )， 而 #define 定义 的 宏 常 量 在 内 存 中 有 若干 个 拷贝 。 














有 因为 





WEN ( 











ノヽ 























只 读 变 量 是 在 编译 的 时 候 确 定 其 值 。 
































\ 有 特定 的 类 型 。 


























1. 11. 3， 修 饰 一 般 变量 


一 般 常量 是 指 简单 类 型 的 只 读 变量 。 这 种 只 读 变量 在 定义 时 ， 修 饰 符 const 可 以 用 在 类 
型 说 明 符 前 ， 也 可 以 用 在 类 型 说 明 符 后 。 例 如 : 


int const i=2; 或 const int i=2; 




















1. 11. 4， 修 饰 数组 
定义 或 说 明 一 个 只 读数 组 可 采用 如 下 格式 : 
int const a[5]={1, 2, 3, 4, 5}: 或 
const int a[5]={1, 2, 3,4,5}; 


1. 11. 5， 修 饰 指针 
const int *p; /p 可変 , p 指向 的 対象 不可 変 
int const *p; /p 可 变 ，p 指向 的 対象 不可 変 
int *const p; /p 不 可 变 ，p 指向 的 対象 可変 
const int *const p; /指针 p 和 op 指向 的 対象 都 不 可変 
在 平时 的 授课 中 发 现 学 生 很 难 记 住 这 几 种 情况 。 这 里 给 出 一 个 记忆 和 理解 的 方法 : 
先 忽 略 类 型 名 (编译 器 解析 的 时 候 也 是 忽略 类 型 名 )， 我 们 看 const 离 哪 个 近 。“ 近 水 楼 
台 先 得 月 ”， 离 谁 近 就 修饰 谁 。 
const int *p; //const 修饰 *p,p 是 指针 ，*p 是 指针 指向 的 对 象 ， 不 可 变 
int-const *p; /const 修饰 *p,p 是 指针 ，*p 是 指针 指向 的 对 象 ， 不 可 变 
int-*const p; //const 修饰 p，p 不 可 变 ，p 指向 的 対象 可変 
const int *const p; // 前 一 介 const 修饰 部 ,后 一 个 const 修饰 p， 指 针 p 和 p 指向 的 対象 
都 不 可 变 


1. 11.6， 修 饰 函数 的 参数 


const 修饰 符 也 可 以 修饰 函数 的 参数 ， 当 不 希望 这 个 参数 值 被 函数 体内 意外 改变 时 使 
Jo PHH: 
void Fun(const int i); 


告诉 编译 器 i 在 函数 体 中 的 不 能 改变 , 从 而 防止 了 使 用 者 的 一 些 无 意 的 或 错误 的 修改 。 


1. 11. 7， 修 饰 函 数 的 返回 值 
const 修饰 符 也 可 以 修饰 函数 的 返回 值 ， 返 回 值 不 可 被 改变 。 例 如 : 


const int Fun (void); 


在 另 一 连接 文件 中 引用 const RERE: 
extern const inti; /正确 的 声明 
extern const intj=10; /错误 ! 只 读 变 量 的 值 不 能 改变 。 


注意 这 里 是 声明 不 是 定义 ， 关 于 声明 和 定义 的 区 别 ， 请 看 本 章 开始 处 。 
讲 了 这 么 多 讲 完了 吗 ? 远 没 有 。 在 CHE, X const 做 了 进一步 的 扩展 ， 还 有 很 多 知识 未 能 
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讲 完 。 























有 兴趣 的 话 ， 不 妨 查找 相关 资料 研究 有 

















1. 12， 最 易 变 的 关键 字 ----volatile 


volatile 是 易 变 的 、 不 稳定 的 意思 。 很 多 人 根本 就 没 见 
也 有 很 多 程序 员 知道 它 的 存在 ， 但 从 来 没 : 




















人 未 识 ” 的 感觉 。 


volatile 关键 字 和 const 一 样 是 一 种 类 型 修饰 









































JE. RIE 





有 


个 关键 字 ， 不 知道 它 的 存在 。 




















未 知 的 因素 更 改 ， 比 如 操作 系统 、 硬 件 或 者 其 它 线程 等 。 遇 有 


译 器 对 访问 该 变量 的 代码 就 不 





先 看 看 下 面 的 例子 : 


inti=10; 

















intj=i: //(1) 语 句 


intk =i; //(2) 语 句 














这 时 候 编 译 器 对 代码 























行 优化 ， 因 














重新 从 内 存 里 取 i 的 人 
ENIT. 
再 看 另 一 个 例子 : 


volatile inti=10: 














intj =i; //(3) 语 句 
intk=i: //(4) 语 句 


， ZP 


FEITH 




















volatile KHET Y Ear i 是 随时 可 能 发 生变 化 的 , 每 次 使 用 它 的 时 候 必须 从 内 存 中 取出 



































提高 了 效率 。 














这 个 值 给 k 赋值 。 




















!“ 杨 家 有 女 初 长 成 , 养 在 深闺 


i 符 , 用 它 修饰 的 变量 表示 可 以 被 某 些 编译 器 

















2) 语句 之 间 i 没有 被 



































| 这 个 关键 字 声明 的 变量 ， 纺 
“化 ， 从 而 可 以 提供 对 特殊 地 址 的 稳定 访问 。 


为 在 (1)、(2) 两 条 语句 中 ，i 没有 被 用 作 左 值 。 这 时 候 
编译 器 认为 i 的 值 没 有 发 生 改 变 ， 所 以 在 (1) 语句 时 从 内 存 中 取 昌 
值 并 没有 被 丢掉 ， 而 是 在 (2) 语句 时 继续 月 


Hi 的 值 赋 给 j 之 后 ， 这 个 
编译 器 不 会 生成 出 汇编 代码 

















JE% 








=. 








的 值 ， 因 而 编译 器 生成 的 汇编 代码 会 重新 从 i 的 地 址 处 读 取 数 据 放 在 k 中 。 
这 样 看 来 ， 如 果 i 是 一 个 寄存 器 变量 或 者 表示 一 个 端口 数据 或 者 是 多 个 线程 的 共享 数 









































据 ， 就 容易 出 错 ， 所 以 说 volatile 可 以 保证 对 特殊 地 址 的 稳定 访问 。 








但 是 注意 : 在 VC++6.0 H 
用 有 可 能 看 不 出 来 。 你 可 以 同时 4 















































PF， 一 般 Debug 模式 没有 进行 代码 优化 ， 所 以 这 个 关键 字 的 作 
E 成 Debug 版 和 Release 版 的 程序 做 个 测试 。 





留 一 个 问题 : const volatile inti=10; 这 行 代 码 有 没有 问题 ? 如 果 没 有 ， 那 i 到底 是 什么 


属性 ? 








1. 13， 最 会 带 帽 子 的 关键 字 一 --extern 





extern， 外 面 的 ` 




















外 来 的 意思 。 














那 它 有 什么 作 


























J? 举 个 例子 : 假设 你 在 大 街 上 看 到 

















一 个 黑 皮肤 绿 眼 睛 红头 发 的 美女 〈 外 星人 ? ) 或 者 帅哥 。 你 的 第 一 反应 就 是 这 人 不 是 国产 
的 。extern 就 相当 于 他 们 的 这 些 区 别 于 中 国人 的 特性 。extern 可 以 置 于 变量 或 者 函数 前 ， 以 
标示 变量 或 者 函数 的 定义 在 别 的 文件 中 ， 下 面 的 代码 用 到 的 这 些 变量 或 函数 是 外 来 的 ， 不 
是 本 文件 定义 的 ， 提 示 编 译 器 遇 到 此 变量 和 函数 时 在 其 他 模块 中 寻找 其 定义 。 就 好 比 在 本 
文件 中 给 这 些 外 来 的 变量 或 函数 带 A 告诉 本 文件 中 所 有 代码 ， 这 些 家 伙 不 是 土著 。 
那 你 想 想 extern 修饰 的 变量 或 函数 是 定义 还 是 声明 ? 






























































































































































看 列子 : 
Ar 文件 中 定义 : B.c 文件 中 用 extern 修饰 : 
inti= 10; extern inti; // 写 成 1= 10; 行 吗 ? 
void fun (void) extern void fun (void); // 两 个 void 可 否 省 略 ? 
{ 
//code 
} 
C.h 文件 中 定义 : D.c 文件 中 用 extern 修饰 
int j =l; extern double j; /这 样 行 吗 ? 为 什么 ? 
int k =2: j=3.0; /这 样 行 吗 ? 为 什么 ? 








至 于 extern “C” RY ++ 的 范畴 , 这 里 就 先 不 讨论 。 当 然 关 于 extern 
的 讨论 还 远 没 有 结束 ， 在 指针 与 数组 那 一 还 会 和 它 亲 密 接触 的 。 




















1. 14，struct 关键 字 























struct 是 个 神奇 的 关键 字 ， 它 将 一 些 相 关联 的 数据 打包 成 一 个 整体 ， 方 便 使 用 。 


在 网 络 协议 、 通 信 控 制 、 嵌 入 式 系统 、 驱 动 开发 等 地 方 ， 我 们 经 常 要 传送 的 不 是 简单 
的 字 节 流 (char 型 数组 )， 而 是 多 种 数据 组 合 起 来 的 一 个 整体 ， 其 表现 形式 是 一 个 结构 体 。 
经 验 不 足 的 开发 人 员 往 往 将 所 有 需要 传送 的 内 容 依 顺序 保存 在 char 型 数组 中 ， 通 过 指针 偏 
移 的 方法 传送 网 络 报 文 等 信息 。 这 样 做 编程 复杂 ， 易 出 错 ， 而 且 一 旦 控制 方式 及 通信 协议 
有 所 变化 ， 程 序 就 要 进行 非常 细致 的 修改 ， 非 常 容易 出 错 。 这 个 时 候 只 需要 一 个 结构 体 就 
能 搞定 。 平 时 我 们 要 求 函 数 的 参数 尽量 不 多 于 4 个， 如果 函 数 的 参数 多 于 4 个 使 用 起 来 非 
常 容易 出 错 ( 包 括 每 个 参数 的 意义 和 顺序 都 容易 弄 错 )， 效率 也 会 降低 (与 具体 CPU FXR, ARM 
芯片 对 于 超过 4 个 参数 的 处 理 就 有 讲究 ， 有 基体 请 参考 相关 资料 )。 这 个 时 候 ， 可 以 用 结构 体 
压缩 参数 个 数 。 






































































































































1. 14. 1， 空 结构 体 多 大 ? 








结构 体 所 占 的 内 存 大 小 是 其 成 
理 那 章 )。 这 点 很 容易 理解 ， 但 是 


所 占 内 存 之 和 《关于 结构 体 的 内 存 对 齐 ， 请 参考 预 处 
摆 的 这 种 情况 呢 ? 


=m 
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struct student 


{ 
}stu; 


sizeof(stu) 的 值 是 多 少 呢 ?在 Visual C++ 6.0 上 测试 一 下 。 











很 遗憾 ， 不 是 0， 而 是 1。 为 什么 呢 ? 你 想 想 ， 如 果 我 们 把 struct student 看 成 一 个 模子 
的 话 ， 你 能 造 出 一 个 没有 任何 容积 的 模子 吗 ? 显然 不 行 。 编 译 器 也 是 如 此 认为 。 编 译 器 认 
为 任何 一 种 数据 类 型 都 有 其 大 小 ， 用 它 来 定义 一 个 变量 能 够 分 配 确定 大 小 的 空间 。 既 然 如 
此 ， 编 译 器 就 理所当然 的 认为 任何 一 个 结构 体 都 是 有 大 小 的 ， 哪 怕 这 个 结构 体 为 空 。 那 万 
一 结构 体 真 的 为 室 ， 它 的 大 小 为 什么 值 比较 合适 呢 ? 假设 结构 体内 只 有 一 个 char 型 的 数据 
成 员 ， 那 其 大 小 为 lbyte《〈 这 里 先 不 考虑 内 存 对 齐 的 情况 ) .也 就 是 说 非 空 结 构 体 类 型 数据 最 
少 需要 占 一 个 字 节 的 空间 ， 而 空 结构 体 类 型 数据 总 不 能 比 最 小 的 非 空 结构 体 类 型 数据 所 占 
的 空间 大 吧 。 这 就 麻烦 了 ， 空 结构 体 的 大 小 既 不 能 为 0， 也 不 能 大 于 1， 怎么 办 ? 定义 为 0.5 
个 byte? 但 是 内 存 地 址 的 最 小 单位 是 1 个 byte, 0.5 个 byte 怎么 处 理 ? 解决 这 个 问题 的 最 好 
办 法 就 是 折 中 ， 编 译 器 理所当然 的 认为 你 构造 一 个 结构 体 数据 类 型 是 用 来 打包 一 些 数据 成 
员 的 ， 而 最 小 的 数据 成 员 需 要 1 个 byte， 编 译 器 为 每 个 结构 体 类 型 数据 至 少 预 留 1 个 byte 
的 空间 。 所 以 ， 空 结构 体 的 大 小 就 定位 1 个 byte。 
































































































































































































































































































































1. 14. 2， 和 柔性 数组 





也 许 你 从 来 没有 听 说 过 柔性 数组 (flexible array) 这 个 概念 ， 但 是 它 确实 是 存在 的 。 
C99 中 ,结构 中 的 最 后 一 个 元 素 允 许 是 未 知 大 小 的 数组 ， 这 就 叫做 柔性 数组 成 员 ， 但 结 
构 中 的 柔性 数组 成 员 前 面 必 须 至 少 一 个 其 他 成 员 。 和 柔性 数组 成 员 人 允许 结构 中 包含 一 个 大 小 可 
变 的 数组 。sizeof 返回 的 这 种 结构 大 小 不 包括 柔性 数组 的 内 存 。 包 含 柔 性 数组 成 员 的 结构 用 
malloc 0 函数 进行 内 存 的 动态 分 配 ， 并 且 分 配 的 内 存 应 该 大 于 结构 的 大 小 ， 以 适应 柔性 数组 
的 预期 大 小 。 
柔性 数组 到 底 如 何 使 用 呢 ? 看 下 面 例子 : 
typedef struct st_type 





























































































































{ 

inti; 

int a[0]; 
}type_a; 














有 些 编译 器 会 报错 无 法 编译 可 以 改 成 ; 
typedef struct st_type 
{ 


Inti; 





























int a[]; 
}type_a; 
这 样 我 们 就 可 以 定义 一 个 可 变 长 的 结构 体 ， 用 sizeofl(type_a) 得 到 的 只 有 4， 就 是 
sizeof(i)=sizeof(int)。 那 个 0 个 元 素 的 数组 没有 占用 空间 ， 而 后 我 们 可 以 进行 变 长 操作 了 。 通 
过 如 下 表达 式 给 结构 体 分 配 内 存 : 
type_a *p = (type_a*)malloc(sizeof(type_a)+100*sizeof(int)); 
































这 样 我 们 为 结构 体 指 针 p 分 配 了 一 块 内 存 。 用 p->item[n] 就 能 简单 地 访问 可 变 长 元 素 。 
但 是 这 时 候 我 们 再 用 sizeof Cp) 测试 结构 体 的 大 小 ， 发 现 仍然 为 4。 是 不 是 很 诡异 ? 我 们 




















不 是 给 这 个 数组 分 配 了 空间 么 ? 


























在 使 用 柔性 数组 时 需要 把 它 当 作 结 构 体 的 一 个 成 员 , 仅 此 而 已 。 
结构 体 没什么 关系 ， 只 是 

需要 说 明 的 是 : C89 不 支持 这 种 东西 
所 支持 的 是 incomplete type， 而 不 是 zero array， 形 同 intitem[0]; 这 
EB 式 是 形 同 intitem[]; 只 不 过 有 些 纺 


扩展 了 ，C99 发 布 之 后 ， 有 些 



































持 的 


C99 发 布 之 前 已 经 有 了 这 种 非 标 准 





别 急 ， 先 回忆 一 下 我 们 前 面 讲 过 的 “模子 ”。 
已 经 确定 不 包含 柔性 数组 的 内 存 大 小 。 和 柔 忆 

















在 定义 这 个 结构 体 的 时 候 ， 模 子 的 大 小 就 
































FE 数组 只 是 编外 人 员 , 不 占 结构 体 的 编制 。 只 是 说 

















“ 挂 羊 头 卖 狗 肉 ” 而 已 ， 
，C99 把 它 作为 一 种 特例 加 入 了 标准 。 但 是 ，C99 









































当然 , 上 面 既 然 用 malloc 函数 分 配 了 内 存 ， 





free(p); 


经 过 上 面 的 讲解 ， 相 信 你 已 经 掌 扩 
掌握 也 无 所 谓 ， 这 个 东西 实在 很 少 用 。 





























1. 14. 3, struct 5 class 的 区 別 


明了 这 个 看 起 来 似乎 很 神秘 的 东西 。 








再 说 白 点 , 柔性 数组 其 实 与 
算 不 得 结构 体 的 正式 成 员 。 



































形式 是 非法 的 ，C99 x 























译 器 把 intitem[0]; 作 为 非 标准 扩展 来 支持 ， 而 且 在 








i 译 器 把 两 者 合 而 为 一 了 。 
肯定 就 需要 用 free 函数 来 释放 内 存 : 









































不 过 实在 要 是 没 











在 C++ 里 struct 关键 字 与 class 关键 字 一 般 可 以 通用 ， 














员 默 认 情况 下 属性 是 public 的 ， 而 class 成 员 却 是 private 的 。 很 多 人 觉得 不 好 记 ， 其 
public 修饰 它 的 成 员 了 











易 。 你 平时 用 结构 体 时 
通用 ， 你 也 不 要 认为 结构 体内 不 能 放 函 数 了 。 


当然 ， 关 于 结构 体 的 讨论 远 没 有 结束 ， 刀 


















































只 有 


个 和 


小 的 区 别 。struct 的 成 
实 很 容 









































吗 ? 既然 struct 关键 字 与 class 关键 字 可 以 

















1. 15，union 关键 字 





union 关键 字 的 用 法 与 struct 的 月 
union 维护 足够 的 空间 来 置 放 多 个 数据 成 员 中 的 “一 种 ”， 而 不 是 为 每 一 个 数据 成 员 配 置 


共用 一 个 空间 ， 


空间 ， 在 union 中 所 有 的 数据 成 员 























有 的 数据 成 员 具 有 相同 的 起 始 地 址 。 例 子 如 下 : 








union StateMachine 


char character: 
int number: 
char *str; 


double exp; 
1; 

















一 个 union 只 配置 
长 度 是 double 型 态 ， 


在 C++ 里 ， 











E 指 针 与 数组 那 一 章 ， 你 还 会 要 和 它 打交道 的 。 


法 非常 类 似 。 















































同一 时 间 只 能 储存 其 中 一 个 数据 成 员 ， 所 











union 的 成 员 默 认 属 性 页 为 public。 union 主要 





个 足够 大 的 空间 以 来 容纳 最 大 长 度 的 数据 成 员 ， 以 上 例 而 言 ， 最 大 
所 以 StateMachine 的 空间 大 小 就 是 double 数据 类 型 的 大 小 。 

















来 压缩 空间 。 如 果 一 些 数据 























不 可 能 在 同一 时 间 同 时 被 用 到 ， 则 可 以 使 用 union。 











1. 15. 1， 大 小 端 模式 对 union 类 型 数据 的 影响 


下 面 再 看 一 个 例子 : 


union 


{ 
inti; 
char a[2]; 
}*p, u; 





p = &u; 

p->a[0] = 0x39; 

p->a[1] = 0x38; 
pi 的 值 应 该 为 多 少 呢 ? 

这 里 需要 考虑 存储 模式 ， 大 端 模式 和 小 端 模 式 。 

大 端 模式 (Big_endian): 字数 据 的 高 字 节 存储 在 低地 址 中 ， 而 字数 据 的 低 字 节 则 存放 
在 高 地 址 中 。 
小 端 模 式 〈Little_endian): 字数 据 的 高 字 节 存储 在 高 地 址 中 ， 而 字数 据 的 低 字 节 则 存放 
在 低地 址 中 。 
union 型 数据 所 占 的 空间 等 于 其 最 大 的 成 员 所 占 的 空间 。 对 union 型 的 成 员 的 存 取 都 是 
相对 于 该 联合 体 基地 址 的 偏 移 量 为 0 处 开始 , 也 就 是 联合 体 的 访问 不 论 对 哪个 变量 的 存 取 都 
是 从 union 的 首 地 址 位 置 开始 。 如 此 一 和 解释， 上面 的 问题 是 否 已 经 有 了 答案 呢 ? 

























































































1. 15. 2， 如 何 用 程序 确认 当前 系统 的 存储 模式 ? 





上 述 问题 似乎 还 比较 简单 ， 那 来 个 有 技术 含量 的 : 请 写 一 个 C 函数 ， 若 处 理 器 是 

Big_endian 的 ， 则 返回 0; 若是 Little_endian 的 ， 则 返回 1。 
先 分 析 一 下 ， 按 照 上 面 关 于 大 小 端 模式 的 定义 ， 假 设 int 类 型 变量 i 被 初始 化 为 1。 
以 大 端 模式 存储 ， 其 内 存 布局 如 下 图 : 












































inti=1; 


高 地 址 











以 小 端 模式 存储 ， 其 内 存 布局 如 下 图 : 








inti=1; 


<— y 
高 地 址 








变量 i 占 4 个 字 节 ， 但 只 有 一 个 字 节 的 值 为 1， 另外 三 个 字 节 的 值 都 为 0。 如 果 取 出 低 






































地 址 上 的 值 为 0， 毫 无 疑问 ， 这 是 大 端 模 式 ， 如 果 取 出 低地 址 上 的 值 为 1， 毫 无 疑问 ， 这 是 

















小 端 模式 。 既 然 如 此 , 我 们 完全 可 以 利用 union 类 型 数据 的 特点 : 所 
到 现在 ， 应 该 知道 怎么 写 了 吧 ? 参考 答案 如 下 : 








int checkSystem( ) 
{ 
union check 
{ 
int i; 
char ch; 
) c; 
ci=l; 


return (c.ch ==1); 


) 











现在 你 可 以 用 这 个 函数 来 测试 你 当前 系统 的 存储 模式 了 。 当 然 你 也 可 以 不 用 函数 而 直 

















接 去 查看 内 存 来 确定 当前 系统 的 存储 模式 。 如 下 图 : 





了 成员 的 起 始 地 址 一 致 。 














图 中 0x01 的 值 存在 低地 址 上， 说 明 当 前 系统 为 小 端 模 式 。 
不 过 要 说 明 的 一 点 是 ， 某 些 系统 可 能 同时 支持 这 两 种 存储 模式 ， 
在 编译 器 的 选项 中 设置 其 存储 模式 。 
留 个 问题 : 
在 x86 系统 下 ， 输 出 的 值 为 多 少 ? 
#include <stdio.h> 
int main() 


( 















































int a[5]={1,2,3,4,5}; 
int *ptrl=(int *)(&a+1); 





你 可 以 用 硬件 跳 线 或 


int *ptr2=(int *)((int)a+1); 
printf("%x,%x",ptrl[-1],*ptr2); 


return 0; 


1. 16, enum 关键 字 














很 多 初学 者 对 枚 举 (enum) 感 到 迷惑 ， 或 者 认为 没什么 用 ， 其 实 枚 举 (enum) 是 个 很 有 用 的 
数据 类 型 。 














一 





1. 16. 1， 枚 举 类 型 的 使 用 方法 


一 般 的 定义 方式 如 下 : 


enum enum_type_name 


{ 
ENUM_CONST_1, 
ENUM_CONST_2, 


ENUM_CONST_n 

} enum_variable_name: 

注意 : enum_type_name 是 自 定 义 的 一 种 数据 数据 类 型 名 ， 而 enum_variable_name 为 
enum_type_name 类 型 的 一 个 变量 , 也 就 是 我 们 平时 常 说 的 枚 举 变量 .实际 上 enum_type_name 
类 型 是 对 一 个 变量 取 值 范围 的 限定 ， 而 花 括 号 内 是 它 的 取 值 范围 ， 即 enum_type_name 类 型 
的 変量 enum_variable_name 只 能 取 值 为 花 括 号 内 的 任何 一 个 值 ， 如 果 赋 给 该 类 型 变量 的 值 
不 在 列表 中 ， 则 会 报错 或 者 警告 。ENUM_CONST_1、ENUM_CONST 2. .... 
ENUM_CONST_n, 这 些 成 员 都 是 常量 , 也 就 是 我 们 平时 所 说 的 枚 举 常量 (常量 一 般 用 大 写 )。 
enum 变量 类 型 还 可 以 给 其 中 的 常量 符号 赋值 ， 如 果 不 赋值 则 会 从 被 赋 初 值 的 那个 常量 开始 
依次 加 1， 如 果 都 没有 赋值 ， 它 们 的 值 从 0 开始 依次 递增 1。 如 分 别 用 一 个 常数 表示 不 同 颜 
色 : 


enum Color 


( 















































= 

































































































































































GREEN = 1, 

RED, 

BLUE, 
GREEN_RED = 10, 
GREEN_BLUE 


}ColorVal; 
其 中 各 常量 名 代表 的 数值 分 别 为 : 




















GREEN = 1 

RED =2 

BLUE = 3 
GREEN_RED = 10 
GREEN_BLUE = 11 


1.16.2, 枚挙 与 #define 宏 的 区 别 


下 面 再 看 看 枚 举 与 #define 宏 的 区 别 : 








宏 常量 是 在 预 9 





1), #define TEY 

















2); 











3)， 枚 举 可 以 一 次 定义 大 量 相 关 的 常量 ， 而 #define 2: 





留 两 个 问题 : 
A)， 枚 举 能 做 到 事 ，#define 宏 能 
B), sizeof (ColorVal) 的 值 为 多 4 





Se 














H 
7E 





不 能 调试 


进行 简单 蔡 换 。 枚 举 常 
般 在 编译 器 里 ， 可 以 调试 枚 举 常 量 ， 但 

















不 能 都 做 到 ? 如 果 能 ， 那 大 


b? 


为 什么 ? 


1. 17， 伟 大 的 缝 幼 师 一 --typedef 关键 字 


1. 17. 1， 关 于 马甲 的 笑话 




















有 这 样 一 个 笑话 : 
到 岸 边 。 猎 人 一 把 抓 住 这 
T! 























个 








号外, 





typedef 关键 字 是 个 伟大 的 缝 幼 师 ， 擅 长 做 马甲 ， 
它 可 以 把 狼 变 成 一 头羊 ， 也 能 把 半 











同样 也 能 把 美丽 的 天 使 变 成 鸟 人 。 
哪 天 我 把 你 当 鸟 人 ， 你 可 别 怪我 。 


个 猎人 在 河 边 抓 捕 











一 条 蛇 ， 蛇 逃 i 
大 声 的 说 道 ; 











量 则 是 在 编译 的 时 候 确 


了 水 里 。 过 一 会 ， 
小 样 ， 别 你 为 你 穿 了 个 马甲 我 就 不 认识 你 




















定 




















= 个 


yqa Je 
































变 成 一 头 狼 。 甚 至 还 可 以 把 长 着 翅膀 的 鸟 人 变 成 天 使 ， 











任何 东西 





穿 上 这 个 马 

















甲 就 立马 变样 。 














ZE rrt 


所 以 ， 你 干 万 不 要 


A A 
— o 





SIEC, 


一 定 要 掌握 它 的 脾气 ， 不 然 


1. 17.2， 历 史 的 误会 -一 -也 许 应 该 是 typerename 











很 多 人 认为 typedef 是 定义 新 的 数 

































































类 型 ， 这 可 能 与 这 个 关键 字 有 关 。 本 来 嘛 ，type 是 




















数据 类 型 的 意思 ; deffine) 是 定义 的 意思 ， 合 起 来 就 是 定义 数据 类 型 啦 。 不 过 很 遗憾 ， 这 种 
理解 是 不 正确 的 。 也 许 这 个 关键 字 该 被 替换 为 “typerename” 或 是 别 的 词 。 
typedef 的 真正 意思 是 给 一 个 已 经 存在 的 数据 类 型 (注意 : 是 类 型 不 是 变量 ) 取 一 个 别 














名 ， 而 非 定义 一 个 新 的 数据 类 型 。 











比如 : 华美 绝伦 的 芍药 ， 就 有 个 别名 --- 





é“ 将 离 “Ny 中 



































代 男 女 交 往 ,往往 以 芍药 相 赠 ,表达 
药 取 了 个 意味 深长 的 别名 





“将 离 ”"。 这 个 





H 1 
































国 古 





情 , 送 刁 药 就 意味 着 即将 分 离 。 所 以 文人 墨客 就 给 区 
新 的 名 字 就 表达 了 那 种 依依 不 金 的 


H| 15... 




















这 样 新 的 名 字 与 原来 的 名 字 相 比 ， 就 更 能 表达 出 想 要 表达 的 意思 。 
在 实际 项 目 中 ， 为 了 方便 ， 可 能 很 多 数据 类 型 (尤其 是 结构 体 之 类 的 自 定义 数据 类 型 ) 
要 我 们 重新 取 一 个 适用 实际 情况 的 别名 。 这 时 候 typedef 就 可 以 帮助 我 们 。 例 如 : 









































typedef struct student 
( 
/code 
}Stu_st,*Stu_pst;// 命 名 规则 请 参考 本 章 前 面部 分 


A), struct student stul; 和 Stu_ststul; 没有 区 别 。 


























B), struct student *stu2; 和 Stu_pststu2; 和 Stu_st *stu2; 没有 区 别 。 

这 个 地 方 很 多 初学 者 迷惑 ，B) 的 两 个 定义 为 什么 相等 呢 ? 其 实 很 好 理解 。 我 们 把 
“struct student [ /*code*/} ”看 成 一 个 整体 ，typedef 跳 是 给 “struct student {/*code*/)" 取 了 个 
别名 叫 “Stu_st” 同时 给 “struct student { /*code*/) *” 取 了 个 别名 叫 “Stu_pst”。 只 不 过 这 两 
个 名 字 同时 取 而 己 ， 好 比 你 给 你 家 小 狗 取 了 个 别名 叫 “ 大 黄 ”， 同 时 你 妹妹 给 小 狗 带 了 小 由 
子 ， 然 后 给 它 取 了 个 别名 叫 “ 小 可 爱 ” A A. 


好 ， 下 面 再 把 typedef 与 const 放 在 一 起 看 看 : 































































































C),const Stu_pst stu3; 

D),Stu_pst const stu4; 

大 多 数 初学 者 认为 C) 里 const 修饰 的 是 stu3 指向 的 対象 ; D〉 Æ const 修饰 的 是 stu4 
这 个 指针 。 很 遗憾 ，C) 里 const 修饰 的 并 不 是 stu3 指向 的 对 象 。 那 const 这 时 候 到 底 修饰 
的 是 什么 呢 ? 我 们 在 讲解 constinti 的 时 候 说 过 const 放 在 类 型 名 “int” 前 后 都 行 ; 而 constint 
*p 5 int * constp 则 完全 不 一 样 。 也 就 是 说 ， 我 们 看 const 修饰 谁 都 时 候 完 全 可 以 将 数据 类 
型 名 视而不见 ， 当 它 不 存在 。 反 过 来 再 看 “const Stu_pststu3” Stu_pst 是 “struct student 
{ /*code*/) 的 列 名 。 “struct student {/*code*/} 关 : 是 一 个 整体 。 对 于 编译 器 来 说 ， 只 认为 
Stu_pst 是 一 个 类 型 名 ， 所 以 在 解析 的 时 候 很 自然 的 把 “Stu_pst” 这 个 数据 类 型 名 忽略 掉 。 
现在 知道 const 到 底 修饰 的 是 什么 了 吧 ? ^ ^。 








































































































1.17.3, typedef 与 #define 的 区 别 





Ha, 上 帝 ! 这 真 要 命 ! 别 急 ， 要 命 的 还 在 后 面 呢 。 看 如 下 例子 : 























E), #define INT32 int 
unsigned INT32 i=10; 
F), typedef int int32; 
unsigned int32 j= 10; 
其 中 P) 编 译 出 错 ， 为 什么 呢 ? E) 不 会 出 错 ， 这 很 好 理解 ， 因 为 在 预 编译 的 时 候 INT32 


被 替换 为 int， 而 unsigned int i= 10; 语句 是 正确 的 。 但 是 ， 很 可 惜 ， 用 typedef 取 的 别 
名 不 支持 这 种 类 型 扩展 。 另 外 ， 想 想 typedef static int int32 行 不 行 ? 为 什么 ? 





































































































下 面 再 看 一 个 与 #define 宏 有 关 的 例子 : 


G), #define PCHAR char* 


























PCHAR p3,p4; 

H), typedef char* pchar; 

pchar pl,p2; 

两 组 代码 编译 都 没有 问题 , 但 是 ,这 里 的 p4 却 不 是 指针 , 仅仅 是 一 个 char 类 型 的 字符 。 
这 种 错误 很 容易 被 忽略 ， 所 以 用 #define 的 时 候 要 慎之 又 愤 。 关 于 #define 当然 还 有 很 多 话题 
需要 讨论 ， 请 看 预 处 理 那 一 章 。 当 然 关 于 typedef 的 讨论 也 还 没有 结束 ， 在 指针 与 数组 那 一 
章 ， 我 们 还 要 继续 讨论 。 




































































1.17.4, #define a int[10]5 typedef int a[10], 


留 两 个 问题 : 

1), #define a int[10] 
A),a[10] al10]: 
B),a[10] a; 
C),int al10]: 


D),int a; 
E),a b[10]; 
F),a b; 


G),a* b[10]: 
H),a* b; 

2), typedef int a[10]: 
A),a[10] al10]: 
B),a[10] a; 
C),int a[10]; 


D),int a; 
E),a b[10]; 
F),a b; 


G),a* b[10]; 


H),a* b; 


3), #definea int*[10] 


A),a[10] al10]: 


B),a[10] 


C),int 
D),int 
E),a 
F),a 
G),a* 


H),a* 


a; 
a[10]; 
a; 

b[10]; 

b; 
b[10]; 


b; 


4), typedef int * a[10]; 


A),a[10] 


a[10]; 


B),a[10] a; 


C),int a[10]; 
D),int a; 
E),a b[10]: 
F),a b; 
G),a* b[10]; 
H),a* b; 

5), #define *a int[10] 
A),a[10] a[10]; 
B),a[10] a; 

C),int a[10]; 

D),int a; 

E),a b[10]; 

F),a b; 

G),a* b[10]; 

H),a* b; 

6),typedefint (* a)[10]; 

A),a[10] al10]: 


B),a[10] a; 


C),int 
D),int 
E),a 


F),a 


a[10]; 
a; 
b[10]; 


b; 


7), #define 


8 


— 


G),a* 


H),a* 


A),a[10] 
B),a[10] 


C),int 
D),int 
E),a 
F),a 
G),a* 


H),a* 


A),a[10] 


b[10]; 

b; 

*a *int[10] 
a[10]; 
a; 
a[10]; 
a; 
b[10]; 
b; 
b[10]; 
b; 


,typedefint* (* a)[10]; 


a[10]; 


B),a[10] a; 


C),int 
D),int 


E),a 




















a[10]; 
a; 
b[10]; 
b; 
b[10]; 























看 哪些 定义 正确 ,哪些 定义 不 正确 。 另 外 ,int[10] 和 a[10] 到 底 该 怎么 月 




















な ロロ 


符号 有 什么 好 说 的 呢 ? 确实 ， 
WEW, 我 问 学 生 : wy 
这 说 明 C 语言 的 基础 掌握 不 牢靠 ， 

















有 














第 二 章 符号 


な ロロ 





符号 可 说 的 内 容 要 少 些 ， 



































这 个 符号 在 C 语言 里 都 
如 果真 正 掌握 























但 总 还 是 有 些 可 以 嘴 明 地 方 。 
用 在 哪些 地 方 ? 没有 一 个 人 能 答 完 整 。 
了 C 语言 ， 你 就 能 很 轻易 的 回答 上 来 


















































































































































名 


o X 


个 问题 就 请 读者 试 着 回答 一 下 吧 。 本 章 不 会 像 关键 字 一 样 一 个 一 个 深入 讨论 ， 只 是 将 容易 
出 错 的 地 方 讨论 一 下 。 
表 (2.1) 标准 C 语言 的 基本 符号 
符号 名 称 符号 == 
ii > ERES 
a. R 
aa | KA 
P: / FH 
i 问号 ` RHL 
| 单 引号 - == 
“ 双 引 号 开本 
I s ) GEES 
[ 左 方 括 号 ] EPHE 
| — ] RRES 
— Ra & and (5) 
A xor CHR) の == 
i ua = 等 于 号 
< ERES " m 
C 语言 的 基本 符号 就 有 20 多 个 ， 每 个 符号 可 能 同时 具有 多 重 含义 ， 而 且 这 些 符号 之 
相互 组 合 又 使 得 C 语言 中 的 符号 变 得 更 加 复杂 起 来 。 





你 也 许 听 说 过 “ 


E| 




















程序 员 。 这 是 他 们 利 











一 个 经 典 作品 : 


#i nclude <stdio.h> 








main(t,_,a)char *a;{return!0<t?t<3?main(-79,-13,a+main(-87,1-_, 


main(-86,0,a+1)+a)):1,t<_?main(t+1,_,a):3,main(-94,-27+t,a)&&t==2?_<13? 


main(2, +1,"%s %d %dm"):9:16:t く 07t く -727main(_,t, 
"@n'+,#/*{ }jwt/w#cdnr/+,{ }r/*de }+,/* {*+,/w{ %+,/w#q#n+,/#{1+,/n {n+,/+#n+,/#\ 
;#q#n+,/+k#;*+,/'r :'d*'3,}{w+K w'K:'+ }e#';dq#'] À 





际 C 语言 乱码 大 赛 (IOCCC)”， 能 获奖 的 人 毫 无 疑问 是 世界 顶级 C 
] C 语言 的 特点 极限 挖掘 的 结果 。 下 面 这 个 例子 就 是 网 上 广 为 流 传 的 


q#'+d'K#!/+k#;q#'rjeKK#)w'rjeKK(nl]'/#.#q#n)()#)w')()(nl]'/+#n'd)rw' i;#\ 

)(nl|!/ní(n#'rí(#w'r nc(nl|'/#(L+ K {rw' iK{;[{nl]/w#q#n'wk nw' \ 

iwk{KK{nl]!l/w{%'l##w#' i; :{nl]/*{q#'1d;r'}{nlwbl!l/*de}'c \ 

;;{n1-{ }rw]/+,}##'* }#nc,,#nw]/+kd'te}+;#'rdq#w! nr7 ") }+}{rl#'{n' DH \ 

} +}##(11/") 

:t<-50?_==*a?putchar(3 1[a]):main(-65,_,a+1):main((*a=='/')+t,_,a+1) 
:0<t?main(2,2,"%s"):*a=='/'||main(0,main(-61,*a, 


"lek;dc i@bK'(q)-[w]* %n+r3#l, { }:\nuwloca-O;m.vpbks,fxntdCeghiry"),a+1);} 





还 没 发 狂 ? 看 来 你 抵抗 力 够 强 的 。 这 是 IOCCC 1988 年 获奖 作品 ， 作 者 是 Ian Phillipps。 
毫 无 疑问 ，Ian Phillipps 是 世界 上 最 顶级 的 C 语言 程序 员 之 一 。 你 可 以 数 数 这 里 面 用 了 多 少 
个 符号 。 当 然 这 里 我 并 不 会 讨论 这 段 代 码 ， 也 并 不 是 鼓励 你 也 去 写 这 样 的 代码 (关于 这 段 代 
码 的 分 析 ， 你 可 以 上 网 查询 )。 恰 恰 相 反 ， 我 要 告诉 你 的 是 : 

大 师 把 代码 写成 这 样 是 经 典 ， 你 把 代码 写成 这 样 是 垃圾 ! 
所 以 在 垃圾 和 经 典 之 间 ， 你 需要 做 一 个 抉择 。 







































































2. 1， 注 释 符号 


2. 1.1， 几 个 似 非 而 是 的 注释 问题 














C 语言 的 注释 可 以 出 现在 C 语言 代码 的 任何 地 方 。 这 人 句 话 对 不 对 ? 这 是 我 当 学 生 时 我 
老师 问 的 一 个 问题 。 我 当时 回答 是 不 对 。 好 ， 那 我 们 就 看 看 下 面 的 例子 : 





















































A), int/*...*/i; 
B), char* s="abcdefgh //hijklmn"; 
C), //Is ita \ 

valid comment? 


D), in/*...*/ti; 























我 们 知道 C 语言 里 可 以 有 两 种 注释 方式 : /* */ 和 //。 那 上 面 3 条 注释 对 不 对 呢 ? 建议 你 
亲自 在 编译 器 中 测试 一 下 。 上 述 前 3 条 注释 都 是 正确 的 ， 最 后 一 条 不 正确 。 


A), 有 人 认为 编译 器 吻 除 掉 注释 后 代码 会 被 解析 成 inti, 所 以 不 正确 。 编译 器 的 确 会 将 六 
释 吻 除 ， 但 不 是 简单 的 剔除 ， 而 是 用 空格 代 栓 原来 的 注释 。 再 看 一 个 例子 : 


记 这 是 */ 机 * 一 条 */define/* 合 法 的 */ID/* 预 处 理 */replacement/* 指 */list/* 令 */ 
你 可 以 用 编译 器 试 试 。 

B), 我 们 知道 双 引 号 引起 来 的 都 是 字符 串 常 量 ， 那 双 斜 杠 也 不 伪 

OQ), 这 是 一 条 合法 的 注释 ， 因 为 \ 是 一 个 接续 符 。 关 于 接续 符 ， 下 面 还 有 更 多 讨论 。 

D), 前 面 说 过 注释 会 被 空格 替换 ， 那 这 条 注释 不 正确 就 很 好 理解 了 。 

现在 你 可 以 回答 前 面 的 问题 了 吧 ? 
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但 注意 : 





庆 ..#/ 这 种 形式 的 注释 不 能 嵌 套 ， 如 ; 





PX a /* JEE J */*/ 











I| AJ s A TJ Pa E A R/E, 


2.1.2, y = x/*p 




















y =x/x*p， 这 是 表示 x 除 以 p 指向 的 内 存 里 的 值 ， 把 结果 赋值 为 y? 我 们 可 以 在 编译 器 


上 测试 一 下 ， 编 译 器 提示 出 错 。 
， 编 译 器 把 /#* 当 作 是 一 段 注释 的 开始 ， 把 /* 后 面 的 内 容 都 当 作 注释 内 容 ， 直 到 出 
现 */ 为 止 。 这 个 表达 式 其 实 只 是 表示 把 x MERA y， 必 后 面 的 内 容 都 当 作 注释 。 但 是 ，| 


实际 上 










































































于 没有 找到 */， 所 以 提示 出 错 。 








我 们 可 以 把 上 面 的 表达 式 修改 一 下 : 
y=x/ *p 

或 者 

y = x/(*p) 


这 样 的 话 ， 表 达 式 的 意思 就 是 x 除 以 p 指向 的 内 存 里 的 值 ， 把 结 
也 就 是 说 只 要 和 斜 枉 O MES C) 之 间 没 有 空格 ， 都 会 被 当 作 注 


— TH >— == 


EIER o 









































2. 1. 3， 怎 样 才能 写 出 出 色 的 注释 


注释 写 得 出 色 非 常 不 容易 ， 但 是 写 得 糟糕 却 是 人 人 可 为 之 。 糟 糕 的 注释 只 会 帮 倒 屏 





2.1.3.1, 





























安息 吧 ， 路 德 维 希 . 凡 . 贝多 芬 





在 《Code Complete》 这 本 书 中 ， 作 者 记录 了 这 样 一 个 故事 : 

















者 已 经 离职 ， 







































































赋值 为 y To 
FE 释 的 开始 。 这 一 点 一 





















































有 位 负责 维护 的 程序 员 半 夜 被 叫 起 来 ， 去 修复 一 个 出 了 问题 的 程序 。 但 是 程序 的 原作 

















没有 办 法 联系 上 他 。 这 个 程序 员 从 未 接触 过 这 个 程序 。 在 仔细 检查 所 有 的 说 明 





后 ， 他 只 发 现 了 一 条 注释 ， 如 下 : 


MOV 


























AX 723h :R.I.P.L.V.B. 













































































































































































这 个 维护 程序 员 通 宵 研究 这 个 程序 ， 还 是 对 注释 百 思 不 得 其 解 。 虽 然 最 后 他 还 是 把 程 
序 的 问题 成 功 排 除了 , 但 这 个 神秘 的 注释 让 他 耿耿 于 怀 。 说 明 一 点 : 汇编 程序 的 注释 是 以 分 
号 开头 。 

几 个 月 后 ， 这 名 程序 员 在 一 个 会 议 上 遇 到 了 注释 的 原作 者 。 经 过 请 教 后 ， 才 明白 这 条 
注释 的 意思 : 安县 吧 ， 路 德 维 希 . 凡 .贝多 芬 (Restin peace, Ludwig Van Neethoven)。 贝 多 芬 于 
1827 年 逝世 ， 而 1827 的 十 六 进 制 正 是 723。 这 真是 让 人 哭笑不得 ! 











2.1.3.2, windows 大 师 们 用 注释 讨论 天 气 问 题 


还 有 个 例子 : 前 些 日 子 windows 的 源 代码 曾经 泄漏 过 一 部 分 





。 人 们 在 





看 这 部 分 大 师 的 




















经 典 作品 时 ， 却 发 现 很 多 与 代码 富 无 关系 的 注释 ! 有 的 注释 在 讨论 天 气 ， 有 的 在 讨论 明天 吃 
什么 , 还 有 的 在 骂 公 司 和 老板 。 这 些 注 释 虽 然 与 代码 无 关 ,， 但 总 比 上 面 那个 让 贝多 芬 安 恩 的 
注释 要 强 些 的 。 至 少 不 会 让 你 抓 狂 。 不 过 这 种 事情 只 有 大 师 们 才 可 以 做 ， 你 可 干 万 别 用 注释 


讨论 天 气 。 
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N = 
























































2. 1. 3. 3， 出 色 注 释 的 基本 要 求 





























【规则 2-1] 注释 应 当 准 确 、 易 懂 ， 防 止 有 二 义 性 。 错 误 的 注释 不 但 无 益 反 而 有 害 。 


【规则 2-2】 边 写 代码 边 注释 ,修改 代码 同时 修改 相应 的 注释 ， 以 保证 注释 与 代码 的 一 致 性 。 
不 再 有 用 的 注释 要 及 时 删除 。 


【规则 2-3】 注 释 是 对 代码 的 “提示 ”， 而 不 是 文档 。 程 序 中 的 注释 应 当 简单 明了 ， 注 释 太 
多 了 会 让 人 眼花 综 乱 。 


【规则 2-4】 一 目 了 然 的 语句 不 加 注释 。 
例如 : i++;/*i 加 1 
多 余 的 注释 

【规则 2-5] 対 干 全局 数 据 (全局 変量 、 常 量 定 又 等 ) 必须 要 加 注释 。 

【规则 2-6】 注 释 采 用 英文 ， 尽 量 避 免 在 注释 中 使 用 缩写 ， 特 别 是 不 常用 缩写 。 

因为 不 一 定 所 有 的 编译 器 都 能 显示 中 文 , 别人 打开 你 的 代码 , 你 的 注释 也 许 是 一 团 乱 

码 。 还 有 ， 你 的 代码 不 一 定 是 懂 中 文 的 人 阅读 。 

【规则 2-7] 注释 的 位 置 应 与 被 描述 的 代码 相 邻 ， 可 以 与 语句 在 同一 行 ， 也 可 以 在 上 行 ， 但 

不 可 放 在 下 方 。 同 一 结构 中 不 同 域 的 注释 要 对 齐 。 

【规则 2-8】 当 代码 比较 长 ， 特 别 是 有 多 重 散 套 时 ， 应 当 在 一 些 段落 的 结束 处 加 注释 ， 便 于 

阅读 。 

【规则 2-9】 注释 的 缩 进 要 与 代码 的 缩 进 一 致 。 

【规则 2-10】 注 释 代码 段 时 应 注重 “为 何 做 “(why)”， 而 不 是 “怎么 做 (how)”。 

说 明 怎 么 做 的 注释 一 般 停 留 在 编程 语言 的 层次 ， 而 不 是 为 了 说 明 问题 。 尽 力 阐 述 “ 怎 么 做 ?” 

的 注释 一 般 没 有 告诉 我 们 操作 的 意图 ， 而 指明 “怎么 做 ”的 注释 通常 是 元 余 的 。 
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【规则 2-11】 数 值 的 单位 一 定 要 注释 。 
注释 应 该 说 明 某 数值 的 单位 到 底 是 什么 意 
米 ， 还 是 干 米 等 ， 关于 时 间 的 必须 说 明 单 位 是 



































思 。 比 如 : 关于 长 度 的 必须 说 明 单位 是 毫米 ， 


Han 
时 ， 分 ， 秒 ， 还 是 毫秒 等 。 
































【规则 2-12】 对 变量 的 范围 给 出 注释 。 

















【规则 2-13】 对 一 系列 的 数字 编号 给 出 注释 ， 尤 其 在 编写 底层 驱动 程序 的 时 候 〈 比 如 管 脚 


编号 )。 












































【规则 2-13】 对 于 函数 的 入 口 出 口 数据 给 出 注释 。 
关于 函数 的 注释 在 函数 那 章 有 更 详细 的 讨论 。 





























2. 2， 接 续 符 和 转 义 符 





C 语言 里 以 反 斜 杜 O 表示 断 行 。 编 译 器 会 将 反 斜 杠 吻 除 掉 ， 跟 在 反 斜 杜 后 面 的 字符 























自动 接续 到 前 一 行 。 但 是 注意 : 








反 斜 杠 之 后 不 能 有 空格 ， 























反 斜 杠 的 下 一 行 之 前 也 不 能 有 空 











格 。 当 然 你 可 以 测试 一 下 加 了 空格 之 后 的 效果 。 我 们 看 看 下 面 的 例子 : 





/这 是 一 条 合法 的 \ 
单行 注释 

















/这 是 一 条 合法 的 单行 注释 











#def\ 








RO 这 是 一 条 合法 的 \ 


宏 定义 


cha\ 





tx s=" 这 是 一 个 合法 的 \\ 


n 字符 串 "; 




















反 斜 杠 除了 可 以 被 用 作 接 续 符 ， 还 能 被 


常用 的 转 义 字符 及 其 含义 : 













































































j 作 转 义 字符 的 开始 标识 。 








转 义 字符 ” 转 义 字 符 的 意义 

wm 回 车 换行 

V 横向 跳 到 下 一 制 表 位 置 

w 竖 向 跳 格 

\b 退 格 

\r 回 车 

V 走 纸 换 页 

W 反 斜 打 符 "W 

\ 单 引 号 符 

\a 鸣 铃 

\ddd 1—3 位 八进制 数 所 代表 的 字符 
whh 1 一 2 位 十 六 进 制 数 所 代表 的 字符 











广义 地 讲 ，C 语言 字符 集中 的 任何 一 个 字符 均 可 用 转 义 字符 来 表示 。 表 中 的 vddd 和 \xhh 











正 是 为 此 而 提出 的 。ddd 和 hh 分 别 为 八 ; 

















\134 表示 反 斜 线 ，\X0A 表示 换行 等 。 





制 和 十 六 进 制 的 ASCII 代码 。 如 \102 表示 字母 "B"， 


2.3， 单 引号 、 双 引号 























还 是 容易 弄 错 这 两 点 。 比 如 : 
个 byte。 关 于 字符 串 常量 在 指针 与 数组 那 章 将 有 更 多 的 讨论 。 


如 : 


2.4 











我 们 知道 双 引 号 引起 来 的 都 是 字符 串 常量 ， 单 引号 引起 来 的 都 是 字符 常量 。 但 初学 者 











“a” 和 “a” 完 全 不 一 样 ， 在 内 存 里 前 者 占 1 个 byte, 后 者 占 2 









































这 两 个 列子 还 好 理解 ， 











看 看 这 三 个 : 








1, ‘1 6, “1” 


第 一 个 是 整形 常数 ，32 
第 二 个 是 字符 常量 ， 占 
第 三 个 是 字符 串 

三 者 表示 的 意义 完全 不 一 样 ， 所 占 的 内 存 大 小 也 不 一 样 ， 初 学 者 往往 弄 错 。 
字符 在 内 存 里 是 以 ASC 

















m 
wu, Es. 
市 里 ， 























‘A “+1。 


> BERN 








位 系统 下 占 4 个 byte; 
1 个 byte: 


占 2 个 byte。 





















































AI 码 存储 的 ， 所 以 字符 常量 可 以 与 整形 常量 或 变量 进行 运 









































和 && 是 我 们 经 常用 到 的 逻辑 运算 符 , 与 按 位 运算 符 | 和 及 是 两 码 事 。 下 一 节 会 介绍 按 位 























运算 符 。 虽 然 简单 ， 但 毕竟 容易 犯错 。 看 例子 : 


inti=0; 
int j=0; 
if((++i>0)||(++j>0)) 
{ 

/打印 出 1 和 j 的 值 。 
} 
结果 :i=1;j=0。 
不 要 惊讶 。 逻 辑 运 算 符 





















































上 两 边 的 条 件 只 要 有 一 个 为 真 ， 其 结果 就 为 真 ， 只 要 有 一 个 结果 


























为 假 ， 其 结果 就 为 假 。if((++i>0 首 (++j>0)) 语 句 中 ， 先 计算 (++i>0)， 发 现 其 结果 为 真 ， 后 面 


的 (++j>0) 便 不 再 计算 。 同 样 
者 注意 。 





























&& 运 算 符 也 要 注意 这 种 情况 。 这 是 很 容易 出 错 的 地 方 ， 希 户 读 




















2. 5， 位 运算 符 


C 语言 中 位 运算 包括 下 面 几 种 : 
& 按 位 与 

| 按 位 或 

^ 按 位 异 或 














前 4 种 操作 很 简单 ,一般 不 会 出 错 。 但 要 注意 按 位 运算 符 | 和 & 与 逻辑 运算 符 | 和 &&& 完 全 
是 两 码 事 ， 别 混淆 了 。 其 中 按 位 异 或 操作 可 以 实现 不 用 第 三 个 临时 变量 交换 两 个 变量 的 值 : 
a ^= b; b A= a;a 人 ^b; 但 并 不 推荐 这 么 做 ， 因 为 这 样 的 代码 读 起 来 很 费劲 。 






















































































2. 5. 1， 左 移 和 右 移 





下 面 讨 论 一 下 左 移 和 右 移 : 

左 移 运算 符 “<<” 是 双 目 运算 符 。 其 功能 把 <<< "左边 的 运算 数 的 各 二 进位 全 部 左 移 若 干 
位 ， 由 “<<” 右 边 的 数 指定 移动 的 位 数 ， 高 位 丢弃 ， 低 位 补 0。 
右 移 运算 符 “>>" 是 双 目 运算 符 。 其 功能 是 把 “>> "左边 的 运算 数 的 各 二 进位 全 部 右 移 若 
干 位 ，“>>” 右 边 的 数 指定 移动 的 位 数 。 但 注意 ;对 于 有 符号 数 ， 在 右 移 时 ， 符 号 位 将 随同 
移动 。 当 为 正 数 时 ， 最 高 位 补 0;， 而 为 负数 时 ， 符 号 位 为 1， 最 高 位 是 补 0 或 是 补 1 取决 
于 编译 系统 的 规定 。Turbo C 和 很 多 系统 规定 为 补 1。 
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2.5.2, 0x01<<2+3 的 值 为 多 少 ? 





再 看 看 下 面 的 例子 : 
0x01<<2+3; 
结果 为 7 吗 ? 测试 一 下 。 结 果 为 32? 别 惊讶 ，32 才 是 正确 答案 。 因 为 “+” 号 的 优先 
级 比 移 位 运算 符 的 优先 级 高 (关于 运算 符 的 优先 级 ， 我 并 不 想 在 这 里 做 过 多 的 讨论 ， 你 几 
乎 可 以 在 任何 一 本 C 语言 书 上 找到 )。 好 ， 在 32 位 系统 下 ， 再 把 这 个 例子 改写 一 下 : 
0x01<<2+30, 或 0x01<<2-3; 
这 样 行 吗 ? 不 行 。 一 个 整 型 数 长 度 为 32 M, 左 移 32 位 发 生 了 什么 事情 ? 溢出 ! 左 移 -1 
位 昵 ? RIKE? 所 以 ， 左 移 和 右 移 的 位 数 是 有 讲究 的 。 左 移 和 右 移 的 位 数 不 能 大 于 数据 
的 长 度 ， 不 能 小 于 0。 









































































































































2. 6， 花 括号 














花 括 号 每 个 人 都 见 过 ， 很 简单 吧 。 但 曾经 有 一 个 学 生 问 过 我 如 下 问题 : 

char a[10] = {“abcde”}; 
他 不 理解 为 什么 这 个 表达 式 正 确 。 我 让 他 继续 改 一 下 这 个 例子 : 

char a[10] { = “abcde”}; 

问 他 这 样 行 不 行 。 那 读者 以 为 呢 ? 为 什么 ? 

花 括号 的 作用 是 什么 呢 ? 我 们 平时 写 函数 ， 让 、while、for、switch 语句 等 都 用 到 了 它 ， 
旦 有 时 又 省 略 掉 了 它 。 简 单 来 说 花 括号 的 作用 就 是 打包 。 你 想 想 以 前 用 花 括 号 是 不 是 为 了 
把 一 些 语 句 或 代码 打 个 包 包 起 来 ， 使 之 形成 一 个 整体 ， 并 与 外 界 绝缘 。 这 样 理解 的 话 ， 上 
面 的 问题 就 不 是 问题 了 。 






















































































==, 















































2.7, ++、 一 操作 符 





这 绝对 是 一 对 让 人 头疼 的 兄弟 。 先 来 点 简单 的 : 


inti= 3; 





(C++i) + Ci) + (++i); 


表达 式 的 值 为 多 少 ? 15 吗 ? 16 吗 ? 18 吗 ? 其 实 对 于 这 种 情况 ，C 语言 标准 并 没有 作出 
规定 。 有 点 编译 器 计算 出 来 为 18， 因 为 i 经 过 3 次 自 加 后 变 为 6， 然 后 3 个 6 相 加 得 18; 

而 有 的 编译 器 计算 出 来 为 16 ( 比 如 Visual C++6.0)， 先 计算 前 两 个 1 的 和 ， 这 时 候 i 自 加 两 
次 , 2 个 i 的 和 为 10， 然 后 再 加 上 第 三 次 自 加 的 i 得 16。 其 实 这 些 没有 必要 辩论 ， 用 到 哪个 
编译 器 写 句 代码 测试 就 行 了 。 但 不 会 计算 出 15 的 结果 来 的 。 


++、-- 作 为 前 缀 ， 我 们 知道 是 先 自 加 或 自 减 ， 然 后 再 做 别 的 运算 ， 但 是 作为 后 缀 时 ， 到 
底 什么 时 候 自 加 、 自 减 ? 这 是 很 多 初学 者 迷糊 的 地 方 。 假 设 =0， 看 例子 : 


A) j =(i++,i++,1++); 

























































































































































































B),for (i=0;i<10;i++) 
{ 
//code 
} 

C), k= G+) + (it+) + (i++); 
你 可 以 试 着 计算 他 们 的 结果 。 
A) 例子 为 逗号 表达 式 ，i 在 遇 到 每 个 喜 号 后 ， 认 为 本 计算 单位 已 经 结束 ，i 这 时 候 自 加 。 

关于 逗号 表达 式 与 “++” 或 “--” 的 连用 ， 还 有 一 个 比较 好 的 例子 : 


1nt x; 






































inti= 3; 


x = (++i, i++, 1+10); 

问 x 的 值 为 多 少 ? i 的 值 为 多 少 ? 

按照 上 面 的 讲解 ， 可 以 很 清楚 的 知道 ， 喜 号 表达 式 中 , i 在 遇 到 每 个 逗号 后 ， 认 为 本 计 
单位 已 经 结束 ，i 这 时 候 自 加 。 所 以 ， 本 例子 计算 完 后 ， i 的 值 为 5，x 的 值 为 15。 
B) 例子 i 与 10 进行 比较 之 后 ， 认 为 本 计算 单位 已 经 结束 ，i 这 时 候 自 加 。 
O) 例子 i 遇 到 分 号 才 认 为 本 计算 单位 已 经 结束 ，i 这 时 候 自 加 。 
也 就 是 说 后 绥 运 算是 在 本 计算 单位 计算 结束 之 后 再 自 加 或 自 减 。C 语言 里 的 计算 
为 以 上 3 类 。 


留 一 个 问题 : 



























































































































































位 大 体 分 

















for (i=0, printf (“First=%d”, i) ; 
i<10, printf (“Second=%d”, i) ; 


i++, printf (“Third=%d”, i)) 


{ 
printf (“Fourth=%d”, i); 
) 
打印 出 什么 结果 ? 





2.7.1, ++i+++i+++i 




















上 面 的 例子 很 简单 ， 那 我 们 把 括号 去 掉 看 看 : 


















































inti=3; 
十 十 1 十 十 十 1 十 十 十 1 
RR! 这 到 底 是 什么 东西 ? 好 ， 我 们 先 看 看 这 个 : a+++b 和 下 面 哪个 表达 式 想 当 : 
A),a++ +b; 
B),a+ ++b; 


2.7.2， 贪 心 法 





C 语言 有 这 样 一 个 规则 : 每 一 个 符号 应 该 包含 尽 可 能 多 的 字符 。 也 就 是 说 ， 编 译 器 将 程 
序 分 解 成 符号 的 方法 是 ， 从 左 到 右 一 个 一 个 字符 地 读 入 ， 如 果 该 字符 可 能 组 成 一 个 符号 ， 
那么 再 读 入 下 一 个 字符 ， 判 断 已 经 读 入 的 两 个 字符 组 成 的 字符 串 是 否 可 能 是 一 个 符号 的 组 
成 部 分 : 如 果 可 能 ， 继 续 读 入 下 一 个 字符 ， 重 复 上 述 判 断 ， 直 到 读 入 的 字符 组 成 的 字符 串 
已 不 再 可 能 组 成 一 个 有 意义 的 符号 。 这 个 处 理 的 策略 被 称 为 “贪心 法 ” 需要 注意 到 是 ， 除 
了 字符 串 与 字符 常量 ， 符 号 的 中 间 不 能 戏 有 空白 〈 空 格 、 制 表 符 、 换 行 符 等 )。 比 如 : == 是 

















































































































单个 符号 ， 而 = = 是 两 个 等 号 。 
按照 这 个 规则 可 能 很 轻松 的 判断 a+++b 表达 式 与 a++ +b 一 致 。 那 ++i+++i+++i; 会 被 解 
析 成 什么 样子 呢 ? 希望 读者 好 好 研究 研究 。 另 外 还 可 以 考虑 一 下 这 个 表达 式 的 意思 : 


a+++++b; 


























2.8, 2/(-2) 的 值 是 多 少 ? 








除法 运算 在 小 学 就 掌握 了 的 ， 这 里 还 要 讨论 什么 昵 ? 别 急 ， 先 计算 下 面 这 个 例子 : 

2/(-2) 的 值 为 多 少 ? 2%(-2) 的 值 呢 ? 

如 果 与 你 想象 的 结果 不 一 致 ， 不 要 惊讶 。 我 们 先 看 看 下 面 这 些 规 则 ; 

假定 我 们 让 a 除 以 b， 商 为 q9， 余 数 为 T: 

q= ab: 

r= a%b; 

这 里 不 妨 先 假定 b 大 于 0. 

我 们 希望 a、b、q、T 之 间 维 持 什 么 样 的 关系 呢 ? 

1， 最 重要 的 一 点 ， 我 们 希望 q*b +r == a， 因 为 这 是 定义 余数 的 关系 。 

2， 如 果 我 们 改变 a 的 正 负 号 ， 我 们 希望 q 的 符号 也 随 之 改变 ， 但 q 的 绝对 值 不 会 变 。 

3, 当 b>0 时 ， 我 们 希望 保证 r>=0 H. r<b。 

这 三 条 性 质 是 我 们 认为 整数 除法 和 余数 操作 所 应 该 具备 的 。 但 是 ， 很 不 幸 ， 它 们 不 可 
能 同时 成 立 。 

先 考 虑 一 个 简单 的 例子 : 3/2， 商 为 1， 余数 也 为 1。 此 时 ， 第 一 条 性 质 得 到 了 满足 。 

好 ， 把 例子 稍微 改写 一 下 : (-3)/2 的 值 应 该 是 多 少 呢 ? 如 果 要 满足 第 二 条 性 质 ， 答 案 应 
该 是 -1。 但 是 ， 如 果 是 这 样 ， 余 数 就 必定 是 -1， 这 样 第 三 条 性 质 就 无 法 满足 了 。 如 果 我 们 首 
先 满足 第 三 条 性 质 ， 即 余数 是 1， 这 种 情况 下 根据 第 一 条 性 质 ， 商 应 该 为 -2， 那 么 第 二 条 性 
质 又 无 法 满足 了 。 

上 面 的 矛盾 似乎 无 法 解决 。 因 此 ，C 语言 或 者 其 他 语言 在 实现 整数 除法 截断 运算 时 ， 必 
须 放 弃 上 述 三 条 性 质 中 的 至 少 一 条 。 大 多 数 编程 语言 选择 了 放弃 第 三 条 , 而 改 为 要 求 余数 与 
被 除数 的 正 负 号 相同 。 这 样 性 质 1 和 性 质 2 就 可 以 得 到 满足 。 大 多 数 C 语言 编译 器 也 都 是 
如 此 。 
























































































































































但 是 ，C 语言 的 定义 只 保证 了 性 质 1， 以 及 当 a>=0 H b>0 时 ， 保 证 |<|b| 以 及 r>=0。 后 
面部 分 的 保证 与 性 质 2 或 性 质 3 比较 起 来 ， 限 制 性 要 弱 得 多 。 
通过 上 面 的 解释 ， 你 是 否 能 准确 算出 27(-2) 和 2%(-2) 的 値 呪 
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2.9， 运 算 符 的 优先 级 


2. 9. 1， 运 算 符 的 优先 级 表 





C 语言 的 符号 众多 , 由 这 些 符 号 又 组 合成 了 各 种 各 样 的 运算 符 。 既 然 是 运算 符 就 一 定 有 
其 特定 的 优先 级 ， 下 表 就 是 C 语言 运算 符 的 优先 级 表 : 















































优先 级 | 运算 符 名 称 或 含义 使 用 形式 结合 方向 说 明 


















































































































































































































































































































































































































































































































































































































































[] 数组 下 标 数组 名 [常量 表达 式 ] 
表达 式 ) /函数 名 ( 形 
1 x b 参 表 ) 左 到 右 
. 成 员 选 择 〈 对 象 ) 对 象 . 成 员 名 
=> 成 员 选 择 〈 指 针 ) | 对 象 指针 -> 成 员 名 
= 负 号 运算 符 -表达 式 単 目 返 算 符 
(类 型 ) 强制 类 型 转换 (数据 类 型 ) 表达 式 
++ 自 增 运算 符 t+ 变量 名 /变量 名 + 単 目 返 算 符 
E 自 减 运算 符 变量 名 /变量 名 单 目 运算 符 
2 * 取 值 运算 符 * 指 针 变量 右 到 左 | 单 目 运算 符 
& 取 地 址 运算 符 & 变 量 名 单 目 运算 符 
! 逻辑 非 运算 符 ! 表 达 式 单 目 运算 符 
- 按 位 取 反 运算 符 “表达 式 单 目 运算 符 
sizeof 长 度 运 算 符 sizeof (表达 式 ) 
Z 除 表达 式 / 表 达 式 双 目 运算 符 
* 乗 表达 式 * 表 达 式 Eat 双 目 运算 符 
TE T | = es a ua 双 目 运算 符 
达 式 
+ 加 表达 式 + 表 达 式 双 目 运算 符 
M 表达 式 -表达 式 | C IU MEZER 
くく 左 移 变量 < 表达 式 双 目 运算 符 
> 右 移 变量 >> 表 达 式 qaa 双 目 运算 符 
> 大 于 表达 式 > 表 达 式 双 目 运算 符 
>= 天 村 等 于 表达 式 >= 表 达 式 左 到 右 双 目 运算 符 
《 小 于 表达 式 《 表 达 式 双 目 运算 符 
<= 小 于 等 于 表达 式 《= 表 达 式 双 目 运算 符 
== 等 于 表达 式 == 表 达 式 EIE 双 目 运算 符 
l= 不 等 于 表达 式 != 表达 式 双 目 运算 符 
8 & 按 位 与 表达 式 & 表 达 式 EIE | 双 目 运算 符 
9 按 位 异 或 表达 式 表达 式 左 到 右 | 双 目 运算 符 
10 | 按 位 或 表达 式 | 表 达 式 左 到 右 | 双 目 返 算 符 
11 && 逻辑 与 表达 式 && 表 达 式 左 到 右 | 双 目 运算 符 
12 || 逻辑 或 表达 式 | | 表达 式 左 到 右 | 双 目 运算 符 
13 | ° gmana C C ta | aae | 三 目 运算 符 
= 赋值 运算 符 变量 = 表达 式 
/= 除 后 赋值 变量 /= 表达 式 
ネー 乘 后 赋值 变量 *= 表 达 式 
%= 取 模 后 赋值 变量 %= 表 达 式 
E MARE TEREA | Pra 
-= 减 后 赋值 变量 -= 表达 式 
¿< 左 移 后 赋值 变量 <= 表 达 式 
>>= 右 移 后 赋值 变量 >>= 表 达 式 

































































< | 按 位 与 后 赋 信 | ”变量 -表达 式 
二 | 按 位 异 或 后 赋值 | ”变量 表达 式 
= “|” 按 位 或 后 赋 信 ”| ”变量 [= 表达 式 
H 页 
15 逗号 运算 符 | 表达 式 , 表达 式 , … | 左 到 右 | SPAM 
序 运算 














注 : 





这 些 东西 ， 











同一 优先 级 的 运算 符 ， 运 算 次 月 
上 表 不 容易 记 住 。 其 实 也 用 不 着 死记 ， 用 得 多 ,看 























己 写 代码 的 











只 要 记 住 乘 除法 的 优先 级 比 加 减法 高 就 行 了 ， 








时 候 , 确实 可 以 , 但 如 果 是 你 去 阅读 和 理 


结合 方向 所 决定 。 


pA 











=] 


已 

















解 别人 的 代码 1 





号 了 吧 ? 所 以 ， 记 住 这 个 表 ， 我 个 人 认为 还 是 很 有 必要 的 。 





2. 9. 2， 一 些 容 易 出 错 的 优先 级 问题 





导 多 自然 就 记得 了 。 也 有 人 说 不 用 记 
别 的 地 方 一 











律 加 上 括号 。 这 在 你 自 
? 别人 不 一 定 都 加 上 括 




















ERP, 优先 级 同 为 1 的 几 种 运算 符 如 果 同 时 出 现 , 那 怎 么 确定 表达 式 的 优先 级 呢 ?” 这 










































































是 很 多 初学 者 迷糊 的 地 方 。 下 表 就 整理 了 这 些 容易 出 错 的 情况 : 
优先 级 问题 表达 式 经 常 误 认 为 的 结果 实际 结果 
. 的 优先 级 高 于 * *p. f p 所 指 対 象 的 字 段 f | XI pH f 偏 移 ， 作 为 
-> 操作 符 用 于 消除 这 (*p). f 指针 ， 然 后 进行 解除 
个 问题 引用 操作 。*(p.f) 
[JAF int *apl] ap 是 个 指向 int 数组 | ap 是 个 元 素 为 int 
的 指针 指针 的 数组 
int (*ap) [] int *(ap[]) 
函数 O 高 于 # int *fp() fp 是 个 函数 指针 ,所 | fp 是 个 函数 ， 返 下 


指 函 数 返 加 
int (*fp) () 





int. 











int * 
int *(fp() ) 





== 和 != 高 于 位 操作 


(val & mask != 0) 


(val & mask)!= 0 


val & (mask != 0) 








== 和 != 高 于 赋值 符 


C 
EOF 


getchar () 


(c = getchar) != 
EOF 


c = (getchar) != 


EOF) 






























































算术 运算 符 高 于 位 移 | msb << 4 + 1sb (msb << 4) + lsb |msb (4 + 1sb) 
运算 符 
逗号 运算 符 在 所 有 运 | i = 1,2 i = (1,2) (i = 1),2 
算 符 中 优先 级 最 低 

这 些 容易 出 错 的 情况 ， 希 望 读者 好 好 在 编译 器 上 调试 调试 ， 这 样 印象 会 深 一 些 。 一 定 要 
多 调试 ， 光 靠 看 代码 ， 水 平 是 很 难 提 上 来 的 。 调 试 代码 才 是 最 长 水 平 的 。 


























往往 我 说 今天 上 课 的 内 容 是 预 处 理 
么 ?这 也 用 得 着 


第 三 章 预 处 理 
































时 ， 便 有 学 生 质 疑 : 预 处 理 不 就 是 include 和 define 

















啊 ? 。 是 的 ， 





于 此 吗 ? 远 远 不 止 。 














先 看 




















非常 值得 讨论 ， 即 使 是 include 和 








几 个 个 常识 性 问题 : 


A), 预 处 理 是 C 语言 的 一 部 分 吗 ? 










































































define。 但 是 预 处 理 仅 限 





























































































































































































































B), 包 含 “#” 号 的 都 是 预 处 理 吗 ? 
C), 预 处 理 指令 后 面 都 不 需要 加 “; “号 吗 ? 
不 要 急 着 回答 ， 先 看 看 ANSI 标准 定义 的 C 语言 预 处 理 指 令 : 
表 (3.1) 预 处 理 指令 

预 处 理 名 称 K Xx 

#define BEL 

#undef 撤销 已 定义 过 的 宏 名 

#include 使 编译 程序 将 男 一 源 文件 骨 入 到 带 有 #include 的 源 文件 中 

#if #if 的 一 般 含义 是 如 果 #if 后 面 的 常量 表达 式 为 true, 则 编译 它 与 #endif 之 

PE 闻 的 代码 ， 和 否则 跳 过 这 些 代码 。 命 令 #endif 标识 一 个 #if 块 的 结束 。#else 
命令 的 功能 有 点 象 C 语言 中 的 else , #else 建立 另 一 选择 〈 在 # 计 失败 

#elif 的 情况 下 )。#elif 命令 意义 与 else if 相同 ， 它 形成 一 个 felse-if 阶梯 状 

#endif 语句 ， 可 进行 多 种 编译 选择 。 

#ifdef Hi#ifdef Ej#ifndef 命令 分 别 表示 “如 果 有 定义 ”及 “如 果 无 定义 ” 是 条 

件 编 译 的 另 一 种 方法 。 

#line 改变 当前 行 数 和 文件 名 称 ， 它 们 是 在 编译 程序 中 预先 定义 的 标识 符 
命令 的 基本 形式 如 下 : 
#line number["filename"] 

#error 编译 程序 时 ， 只 要 遇 到 #error 就 会 生成 一 个 编译 错误 提示 消息 ， 并 停止 
编译 

#pragma 为 实现 时 定义 的 命令 , 它 允 许 向 编译 程序 传送 各 种 指令 例如 , 编译 程序 可 





能 有 一 种 选择 
跟踪 选择 。 
































它 文 持 对 程序 执行 的 跟踪 。 可 

















J#pragma 语句 指定 一 个 





另外 ANSI 标准 C 还 定义 了 如 下 几 个 宏 : 


_LINE_ 表示 了 











E 在 编译 的 文件 的 行 号 


FILE. 表示 正在 编译 的 文件 的 名 字 








_DATE_ 表示 编译 时 刻 的 日 期 字符 串 ， 例 如 : "25 Dec 2007" 

_TIME_ 表示 编译 时 刻 的 时 间 字 符 串 ， 例 如 : "12:30:55" 

_STDC_ 判断 该 文件 是 不 是 定义 成 标准 C 程序 

如 果 编 译 器 不 是 标准 的 ， 则 可 能 仅 支持 以 上 宏 的 一 部 分 ， 或 根本 不 支持 。 当 然 编译 器 
也 有 可 能 还 提供 其 它 预定 义 的 宏 名 。 注 意 : 宏 名 的 书写 由 标识 符 与 两 边 各 二 条 下 划 线 构成 。 

相信 很 多 初学 者 ,甚至 一 些 有 经 验 的 程序 员 都 没有 完全 掌握 这 些 内 容 , 下 面 就 一 一 详细 
讨论 这 些 预 处 理 指 令 。 





















































































































































3. 1. 1， 数 值 宏 常 量 


#define 宏 定义 是 个 演技 非常 高 超 的 蔡 身 演员 ， 但 也 会 经 常 机 大 有 牌 的 ， 所 以 我 们 用 它 要 
慎之 又 慎 。 它 可 以 出 现在 代码 的 任何 地 方 ， 从 本 行 宏 定义 开始 ， 以 后 的 代码 就 就 都 认识 这 
个 宏 了 ; 也 可 以 把 任何 东西 定义 成 宏 。 因 为 编译 器 会 在 预 编译 的 时 候 用 真 身 替换 蔡 身 ， 而 
在 我 们 的 代码 里 面 却 又 用 常常 用 蔡 身 来 帮忙 。 看 例子 : 


#define PI 3.141592654 


在 此 后 的 代码 中 你 尽 可 以 使 用 PI 来 代 替 3.141592654, 而且 你 最 好 就 这 么 做 。 不 然 的话 ， 如 
果 我 要 把 PI 的 精度 再 提高 一 些 ， 你 是 否 愿 意 一 个 一 个 的 去 修改 这 串 数 呢 ? 你 能 保证 不 漏 不 
出 错 ? 而 使 用 PI 的 话 ， 我 们 却 只 需要 修改 一 次 。 这 种 情况 还 不 是 最 要 命 的 ， 我 们 再 看 一 个 
例子 : 

#define ERROR POWEROFF -1 


如 果 你 在 代码 里 不 用 ERROR_POWEROFF 这 个 宏 而 用 -1， 尤 其 在 函数 返回 错误 代码 的 时 
(往往 一 个 开发 一 个 系统 需要 定义 很 多 错误 代码 )。 肯 怕 上 帝都 无 法 知道 -1 表示 的 是 什么 
思 吧 。 这 个 -1， 我 们 一 般 称 为 “魔鬼 数 ” 上 帝 遇 到 它 也 会 发 狂 的 。 所 以 ， 我 奉劝 你 代码 上 
一 定 不 要 出 现 “魔鬼 数 ” 

第 一 章 我 们 详细 讨论 了 const 这 个 关键 字 ， 我 们 知道 const 修饰 的 数据 是 有 类 型 的 ， 而 
define 宏 定 义 的 数据 没有 类 型 。 为 了 安全 ， 我 建议 你 以 后 在 定义 一 些 宏 常 数 的 时 候 用 const 
代替 ， 编 译 器 会 给 const 修饰 的 只 读 变 量 做 类 型 校 验 , 减少 错误 的 可 能 。 但 一 定 要 注意 const 
修饰 的 不 是 常量 而 是 readonly 的 变量 ,const 修饰 的 只 读 变量 不 能 用 来 作为 定义 数组 的 维 数 ， 
也 不 能 放 在 case 关键 字 后 面 。 
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3. 1.2， 字 符 串 宏 常量 























除了 定义 宏 常 数 之 外 ， 经 常 还 用 来 定义 字符 串 ， 尤 其 是 路 径 : 
A),#define ENG_PATH 1 BE:uEnglishMisten_to_thisMisten_to_this_3 


B)#define ENG PATH 2 “E:\English\isten_to_this\listen_to_this_3” 








噢 ， 到 底 哪 一 个 正确 呢 ? 如 果 路 径 太 长 ， 一 行 写 下 来 比较 别扭 怎么 办 ?用 反 斜 杠 接续 

















C), #define ENG PATH 3 E:\English\listen_to_this\listen\ 

_to_this 3 

还 没 发 现 问题 ? 这 里 用 了 4 个 反 斜 杜 ， 到 底 哪个 是 接续 符 ? 回去 看 看 接续 符 反 和 斜 杠 。 
反 斜 杜 作为 接续 符 时 ， 在 本 行 其 后 面 不 能 再 有 任何 字符 ， 空 格 都 不 行 。 所 以 ， 只 有 最 后 一 
个 反 和 斜 杠 才 是 接续 符 。 至 于 A) 和 B)， 那 要 看 你 怎么 用 了 ， 既 然 define Z H jé fiA 
那 给 ENG_PATH_1 加 上 双 引 号 不 就 成 了 :“ENG_PATH_ 1?。 

但 是 请 注意 : 有 的 系统 里 规定 路 径 的 要 用 双 反 和 斜 枉 “\N”, 比 如 : 

#define ENG PATH 4 E:\\English\\listen_to_this\\listen_to_this_3 
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3.1.3, 用 define 宏 定义 注释 符号 ? 

















EHX} define 的 使 用 都 很 简单 ， 再 看 看 下 面 的 例子 ; 
#define BSC // 




















#define BMC /* 

#define EMC */ 

D),BSC my single-line comment 
E),BMC my multi-line comment EMC 


D) 和 EE) 都 错误 ， 为 什么 呢 ?” 因 为 注释 先 于 预 处 理 指令 被 处 理 , 当 这 两 行 被 展开 成 //... 或 
访 ...*/ 时 ,注释 已 处 理 完毕 ,此 时 再 出 现 //... 或 /*...*/ 自 然 错 误 .因此 ,试图 用 宏 开始 或 结束 一 段 
注释 是 不 行 的 。 







































































3.1.4, 用 define 宏 定 义 表 达 式 


这 些 都 好 理解 ， 下 面 来 点 有 “技术 含量 ”的 ; 

定义 一 年 有 多 少 秒 : 

#define SEC A_YEAR 60*60*24*365 

这 个 定义 没 错 吧 ? 很 遗憾 ， 很 有 可 能 错 了 ， 至 少 不 可 靠 。 你 有 没有 考虑 在 16 位 系统 下 
把 这 样 一 个 数 赋 给 整 型 变量 的 时 候 可 能 会 发 生 溢 出 ? 一 年 有 多 少 秒 也 不 可 能 是 负数 吧 。 修 
改 一 下 : 
#define SEC A_YEAR (60*60*24*365) UL 
又 出 现 一 个 问题 ， 这 里 的 括号 到 底 需 不 需要 呢 ? 继续 看 一 个 例子 : 

定义 一 个 宏 函 数 ， 求 x 的 平方 : 

































































#define SQR(x) x*x 
对 不 对 ?” 试 试 : 假设 x 的 值 为 10，SQR (CO 被 替换 后 变 成 10*10。 没 有 问题 。 


了 试 试 : 假设 x 的 值 是 个 表达 式 10+1, SOR (x) 被 替 換 后 変成 10+1*10+1。 问 题 来 了 ， 
这 并 不 是 我 想 要 得 到 的 。 怎 么 办 ? 括号 括 起 来 不 就 完了 ? 


#define SQR(x) (Cx) * (x)) 
最 外 层 的 括号 最 好 也 别 省 了 ， 看 例子 : 
求 两 个 数 的 和 : 
#define SUM (x) (x) + (x) 



































"tal 



























































如果 x 的 值 是 个 表达 式 $*3, 而 代码 又 写成 这 样 :SUM (x)* SUM (x)。 替 換 后 変成 :(5*3)+ 
(5*3) * (5*3) + 《5*3)。 又 错 了 ! 所 以 最 外 层 的 括号 最 好 也 别 省 了 。 我 说 过 define 是 个 
演技 高 超 的 替身 演员 ， 但 也 经 常 机 大 牌 。 要 搞定 它 其 实 很 简单 ， 别 音 甫 括号 就 行 了 。 
注意 这 一 点 : 宏 函 数 被 调用 时 是 以 实 参 代 换 形 参 。 而 不 是 “ 值 传送 ”。 
四 个 问题 : 


A)， 上 述 宏 定 义 中 “SUM”“SQR” 是 宏 吗 ? 















































HS 








B), #define EMPTY 
这 样 定义 行 吗 ? 
C)， 打 印 上 述 宏 定 义 的 值 ，printf “SUM (x); 结果 是 什么 ? 


D), “#define M 100” 是 宏 定义 吗 ? 


3. 1. 5， 宏 定义 中 的 空格 





另外 还 有 一 个 问题 需要 引起 注意 ， 看 下 面 例子 : 

#define SUM (x) (x) + (x) 

这 还 是 定义 的 宏 函 数 SUM (x) HB? 显然 不 是 。 编 译 器 认为 这 是 定义 了 一 个 宏 : SUM, 
其 代表 的 是 (x) (x) + (x)。 为 什么 会 这 样 呢 ? 其 关键 问题 还 是 在 于 SUM 后 面 的 这 个 空 
格 。 所 以 在 定义 宏 的 时 候 一 定 要 注意 什么 时 候 该 用 空格 ,什么 时 候 不 该 用 空格 。 这 个 空格 仅 
仅 在 定义 的 时 候 有 效 ， 在 使 用 这 个 宏 函 数 的 时 候 ， 空 格 会 被 编译 器 忽略 掉 。 也 就 是 说 ， 上 
节 定 义 好 的 宏 函 数 SUM (x) 在 使 用 的 时 候 在 SUM 和 (x) 之 间 留 有 空格 是 没 问题 的 。 比 
如 : SUM (3) 和 SUM (3) 的 意思 是 一 样 的 。 














































































































3.1.6, #undef 




















#undef 是 用 来 撤销 宏 定义 的 ， 用 法 如 下 : 
#define PI 3.141592654 








// code 


#undef PI 
/下 面 的 代码 就 不 能 用 PI 了 ， 它 已 经 被 撤销 了 安定 义 。 
也 就 是 说 宏 的 生命 周期 从 #define 开始 到 #undef 结束 。 很 简单 ， 但 是 请 思考 一 下 这 个 问题 ; 


#define X 3 
































#define Y X*2 
#undef X 
#define X 2 
int z=Y; 


z 的 值 为 多 少 ? 


3. 2， 条 件 编 译 














条 件 编译 的 功能 使 得 我 们 可 以 按 不 同 的 条 件 去 编译 不 同 的 程序 部 分 , 因而 产生 不 同 的 目 
标 代码 文件 。 这 对 于 程序 的 移植 和 调试 是 很 有 用 的 。 条 件 编译 有 三 种 形式 ， 下 面 分 别 介绍 : 

第 一 种 形式 : 

#ifdef 标识 符 

程序 段 1 

#else 

程序 段 2 

#endif 
它 的 功能 是 ， 如 果 标 识 符 已 被 #define 命令 定义 过 则 对 程序 段 1 进行 编译 ;否则 对 程序 段 2 
进行 编译 。 如 果 没 有 程序 段 2( 它 为 空 )， 本 格式 中 的 #else 可 以 没有 ， 即 可 以 写 为 : 

#ifdef 标识 符 

程序 段 

#endif 
第 二 种 形式 : 

#ifndef 标识 符 

程序 段 1 

#else 

程序 段 2 

#endif 
与 第 一 种 形式 的 区 别 是 将 “ifdef* 改 为 “ifndef*。 它 的 功能 是 ， 如 果 标 识 符 未 被 #define 命令 定 
义 过 则 对 程序 段 1 进行 编译 ， 否 则 对 程序 段 2 进行 编译 。 这 与 第 一 种 形式 的 功能 正 相 反 。 
第 三 种 形式 : 


#if 常量 表达 式 






































































































































































































































#else 
程序 段 2 
#endif 





它 的 功能 是 ， 如 常量 表达 式 的 值 为 真 ( 非 0)， 则 对 程序 段 1 进行 编 























isi 




















nÆ. 因此 可 以 使 程序 在 不 同 








ス 、 へ >= 
命令 意 


至 于 #elif 


3.3, 文件 包含 





文件 包含 是 预 处 型 








义 与 elseif 相 





条 件 


ト 








, 完成 不同 的 功能 。 
同 , 它 形成 一 个 if else-if 阶梯 状语 句 , 可 进行 多 种 编译 选择 。 





的 一 个 习 








全 


宏 替 换 的 延伸 ， 有 两 和 
格式 1: 


#include <filename> 



































格式 : 


rB, filename 为 要 包含 的 文件 名 称 ， 用 尖 
系统 规定 的 路 径 中 去 获得 这 个 文件 〈 即 C 2 


要 功能 ， 它 可 月 





























括号 括 起 来 ， 




















文件 )。 找 到 文件 后 ， 用 文件 内 容 蔡 换 该 语句 。 


格式 2: 


#include “filename” 





























中 , filename 为 要 包含 的 文件 名 称 。 双 引 





























译 ， 和 否则 对 程序 段 2 进行 


























来 把 多 个 源 文 件 连接 成 一 个 源 文 件 进行 编 
译 ， 结 果 将 生成 一 个 目标 文件 。C 语言 提供 ##include 命令 来 实现 文件 包含 的 操作 ， 它 实际 是 





























译 系统 所 提供 的 ; 




















号 表示 预 处 























filename 的 文件 ， 若 没有 找到 ， 则 按 系 统 指 定 的 路 径 信息 ， 


文件 内 容 蔡 换 该 语句 。 








需要 强调 的 一 点 是 ，#iinclude 是 将 已 存在 文件 的 内 容 嵌 入 到 当前 文件 中 。 

















搜索 其 他 目 























也 称 为 头 文件 ， 表 示 预 处 理 到 
存放 在 指定 的 子 目 录 下 的 头 





里 应 在 当前 目录 中 查找 文件 名 为 
Ko RACH 





E 后 , 用 








男 外 关于 #include 的 路 径 也 有 点 要 说 明 : include 支持 相対 路 径 , 格式 如 trackant( 蚁 迹 寻 


EDIT: 




















.代表 当前 目 








I 
>K 





3.4, Herror 预 


IÑ 





处 理 

































































#error 预 处 理 指令 的 作用 是 ， 编 译 程 序 时 ， 只 要 遇 到 #error 就 会 生成 一 个 编译 
示 消 息 ， 并 停止 编译 。 其 语法 格式 为 : 


























#error error-mess age 


注意 , 宏 車 error-message 不 用 双 引 号 包 











还 显示 乡 

















有 译 程序 作者 预 2 














相关 资料 ， 这 里 不 浪费 篇 幅 来 做 讨论 。 








错误 提 








围 。 遇 到 #error 指令 时 ， 错 误 信 息 被 显示 ， 可 能 同时 
定义 的 其 他 内 容 。 关 于 系统 所 文 持 的 error-message 信息 ， 请 





查找 

















3.5, #1ine 预 处 理 





























#line 的 作用 是 改变 当前 行 数 和 文件 名 称 ， 它 们 是 在 编译 程序 中 预先 定义 的 标识 符 
命令 的 基本 形式 如 下 : 

#line number["filename"] 
其 中 口内 的 文件 名 可 以 省 略 。 
例如 : 

#line 30 a.h 


其 中 ， 文件 名 ah 可 以 省 略 不 写 。 

这 条 指令 可 以 改变 当前 的 行 号 和 文件 名 , 例如 上 面 的 这 条 预 处 理 指令 就 可 以 改变 当前 的 行 号 
为 30， 文 件 名 是 ah。 初 看 起 来 似乎 没有 什么 用 ， 不 过 ， 他 还 是 有 点 用 的 ， 那 就 是 用 在 编译 
器 的 编写 中 ， 我 们 知道 编译 器 对 C 源码 编译 过 程 中 会 产生 一 些 中 间 文 件 ， 通 过 这 条 指令 ， 
可 以 保证 文件 名 是 固定 的 ， 不 会 被 这 些 中 间 文 件 代替 ， 有 利于 进行 分 析 。 









































































































































3.6, #pragma 预 处 理 




















在 所 有 的 预 处 理 指令 中 ，#pragma 指令 可 能 是 最 复杂 的 了 ， 它 的 作用 是 设 定 编译 器 的 
状态 或 者 是 指示 编译 器 完成 一 些 特定 的 动作 。#pragma 指令 对 每 个 编译 器 给 出 了 一 个 方法 ， 
在 保持 与 C 和 C ++ 语言 完 全 兼容 的 情况 下 ,给 出 主机 或 操作 系统 专 有 的 特征 。 依 据 定义 ,编译 
指示 是 机 器 或 操作 系统 专 有 的 , 且 对 于 每 个 编译 器 都 是 不 同 的 。 

其 格式 一 般 为 : 


#pragma para 


其 中 para 为 参数 ， 下 面 来 看 一 些 常用 的 参数 。 




























































































3.6.1, Hpragma message 


message 参数 : Message SA ERREK- DER C Be 6 (E EAE E H 
口中 输出 相应 的 信息 ， 这 对 于 源 代码 信息 的 控制 是 非常 重要 的 。 其 使 用 方法 为 : 

#pragma message(“ 消 息 文本 ”) 

当 编 译 器 遇 到 这 条 指令 时 就 在 编译 输出 窗口 中 将 消息 文本 打印 出 来 。 
当 我 们 在 程序 中 定义 了 许多 宏 来 控制 源 代码 版 本 的 时 候 ， 我 们 自己 有 可 能 都 会 忘记 有 没有 
正确 的 设置 这 些 宏 ， 此 时 我 们 可 以 用 这 条 指令 在 编译 的 时 候 就 进行 检查 。 假 设 我 们 希望 类 
断 自己 有 没有 在 源 代码 的 什么 地 方 定义 了 _X86 这 个 宏 可 以 用 下 面 的 方法 

#ifdef X86 

#Pragma message(“_X86 macro activated!”) 

#endif 

当 我 们 定义 了 _X86 这 个 宏 以 后 ， 应 用 程序 在 编译 时 就 会 在 编译 输出 窗口 里 显示 “_ 
X86 macro activated!”"。 我 们 就 不 会 因为 不 记得 自己 定义 的 一 些 特定 的 宏 而 抓 耳 找 腮 了 














Er. 
E 
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3.6.2, Hpragma code seg 
得 比较 多 的 pragma 参数 是 code_seg。 格 式 如 : 


#pragma code_seg( ["section-name"[,"section-class"] ] ) 


男 一 个 使 月 




















它 能 够 设置 程序 中 函数 代码 存放 的 代码 段 ， 当 我 们 开发 驱动 程序 的 时 候 就 会 使 用 到 它 。 


3.6.3, #Dragma once 


#pragma once (比较 常用 ) 





Visual C++6.0 中 就 已 经 有 了 ， 但 是 考虑 到 兼容 性 


只 要 在 头 文 件 的 最 开始 加 入 这 








条 指 


令 就 能 够 























3.6.4, #Dragma hdrstop 


#pragma hdrstop 表示 预 编 译 头 文件 到 此 为 止 ， 后 面 的 基文 人 
了 预 编译 头 文件 以 加 快 链接 的 速度 ， 但 如 果 所 有 头 文 件 
所 以 使 用 这 个 选项 排除 一 些 头 文件 。 























有 时 













































































元 之 间 有 依赖 关系 ， 比 如 单元 A 依頼 







































































都 进行 预 编译 又 可 能 占 太 多 磁盘 空 











和 元 B， 所 以 单元 B 


保证 头 文件 被 编译 一 次 ， 这 条 指令 实际 上 在 
并 没有 太 多 的 使 用 它 
F 不 进行 预 编译 。BCB 可 以 











间 ， 


要 先 于 单元 A 编译 。 



























































































































































你 可 以 用 #pragma startup 指定 编译 优先 级 ， 如 果 使 用 了 #pragma package(smart_init) , BCB 
就 会 根据 优先 级 的 大 小 先后 编译 。 
3.6.5, #pragma resource 

#pragma resource "*.dfm" 表 示 把 *.dfm 文件 中 的 资源 加 入 工程 。*.dfm 中 包括 窗 体 
外 观 的 定义 。 
3.6.6, #Dragma warning 

#pragma warning( disable : 4507 34; once : 4385; error : 164 ) 

等 价 于 : 

#pragma warning(disable:4507 34) // 不 显示 4507 和 34 号 警告 信息 

#pragma warning(once:4385) // 4385 号 警告 信息 仅 报 告 一 次 

#pragma warning(error:164) / 把 164 号 警告 信息 作为 一 个 错误 。 

同时 这 个 pragma warning 也 支持 如 下 格式 : 

#pragma warning( push [ ,n ] ) 

#pragma warning( pop ) 

这 里 n 代表 一 个 警告 等 级 (1---4)。 

#pragma warning( push ) 保 存 所 有 警告 信息 的 现 有 的 警告 状态 

#pragma warning( push, mn) 保 存 所 有 警告 信息 的 现 有 的 警告 状态 ， 并 且 把 全 局 警告 
等 级 设 定 为 n。 

#pragma Warning( pop ) 向 栈 中 弹出 最 后 一 个 警告 信息 ， 在 入 栈 和 出 栈 之 间 所 作 的 


一 切 改动 取消 。 例 如 : 
#pragma warning( push ) 


#pragma warning( disable : 4705 ) 


#pragma warning( disable : 4706 ) 
#pragma warning( disable : 4707 ) 


#pragma warning( pop ) 
在 这 段 代 码 的 最 后 ， 重 新 保存 所 有 的 警告 信息 (包括 4705, 




















3.6.7, #Dragma comment 


者 pragma comment(.…) 
该 指令 将 一 个 注释 记录 放 入 一 个 对 象 文件 或 可 执行 文件 中 
常用 的 lib 关键 字 ， 可 以 帮 我 们 连 入 一 个 库 文件 。 比如 : 
#pragma comment(lib, "user32.hb") 


该 指令 用 来 将 user32.lib 库 文件 加 入 到 本 工程 中 。 


















































4706 和 4707)。 


o 


linker: 将 一 个 链接 选项 放 入 目标 文件 中 ,你 可 以 使 用 这 个 指令 来 代替 由 命令 行 传 入 的 或 




















者 在 开发 环境 中 设置 的 链接 选项 ,你 可 以 指定 /include 选项 来 强 














#pragma comment(linker, "/include:_ mySymbol'") 


3.6.8, Hpragma pack 














这 里 重点 讨论 内 存 对 齐 的 问题 和 #pragmapack〈) 的 使用 方 法 。 
什么 是 内 存 对 齐 ? 
先 看 下 面 的 结构 : 


struct TestStruct1 


( 


char c1; 

















short s; 
char c2; 
inti; 
9 
假设 这 个 结构 的 成 员 在 内 存 中 是 紧凑 排列 的 , 假设 cl 的 地 




















地 址 为 00000003,i 地 址 为 00000004。 
可 是 ， 我 们 在 Visual C++6.0 中 写 一 个 简单 的 程序 : 


struct TestStructl a: 








prmtf("c1 %p, s %p, c2 %p, i %p\n", 
(unsigned int)(void*)&a.cl - (unsigned int)(void* )&a, 
(unsigned int)(void*)&a.s - (unsigned int)(void*)&a, 
(unsigned int)Xvoid*)&a.c2 - (unsigned int)(void* )&a, 
(unsigned int)(void*)&a.i - (unsigned int)(void*)&a); 
运行 ， 输 出 : 
c1 00000000, s 00000002, c2 00000004, i 00000008。 


EEEN 2 A: 





址 是 0, 那么 s 的 地 址 就 应 该 





是 1, c2 的 地 址 就 是 3, i 的 地 址 就 是 4。 也 就 是 cl 地 址 为 00000000,s 地 址 为 00000001,c2 


为 什么 会 这 样 ? 这 就 是 内 存 对 齐 而 导致 的 问题 。 





3. 6. 8. 1， 为 什么 会 有 内 存 对 齐 ? 

















字 ， 双 字 ， 和 四 字 在 自然 边界 上 不 需要 在 内 存 中 对 齐 。( 对 字 ， 双 字 ， 和 四 字 来 说 ， 自 
然 边界 分 别 是 侦 数 地 址 ， 可 以 被 4 整除 的 地 址 ， 和 可 以 被 8 整除 的 地 址 。) 无论 如 何 ， 为 了 
提高 程序 的 性 能 ， 数 据 结构 (尤其 是 栈 ) 应 该 尽 可 能 地 在 自然 边界 上 对 齐 。 原 因 在 于 ， 为 
了 访问 未 对 齐 的 内 存 ， 处 理 咒 需要 作 两 次 内 存 访问 ， 然而， 对齐 的 内 存 访问 仅 需要 一 次 访 
问 。 







































































一 个 字 或 双 字 操 作 数 跨越 了 4 字 节 边界 ， 或 者 一 个 四 字 操 作 数 跨越 了 8 字 节 边界 ， 被 
认为 是 未 对 齐 的 ， 从 而 需要 两 次 总 线 周 期 来 访问 内 存 。 一 个 字 起 始 地址 是 奇数 但 却 没有 跨 
越 字 边 界 被 认为 是 对 齐 的 ， 能 够 在 一 个 总 线 周 期 中 被 访问 。 某 些 操作 双 四 字 的 指令 需要 内 
存 操作 数 在 自然 边界 上 对 齐 。 如 果 操 作 数 没 有 对 齐 ， 这 些 指令 将 会 产生 一 个 通用 保护 异常 。 
双 四 字 的 自然 边界 是 能 够 被 16 整除 的 地 址 。 其 他 的 操作 双 四 字 的 指令 允许 未 对 齐 的 访问 
(不 会 产生 通用 保护 异常 )， 然 而 ， 需 要 额外 的 内 存 总 线 周期 来 访问 内 存 中 未 对 齐 的 数据 。 


缺 省 情况 下 ， 编 译 器 默认 将 结构 、 栈 中 的 成 员 数 据 进行 内 存 对 齐 。 因 此 ， 上 面 的 程序 输 
出 就 变 成 了 : cl 00000000, s 00000002, c2 00000004, i 00000008。 编译 器 将 未 对 齐 的 成 员 向 后 
移 ， 将 每 一 个 都 成 员 对 齐 到 自然 边界 上 ， 从 而 也 导致 了 整个 结构 的 尺寸 变 大 。 尽 管 会 牺牲 
一 点 空间 (成 员 之 间 有 部 分 内 存 空间 )， 但 提高 了 性 能 。 也 正 是 这 个 原因 ， 我 们 不 可 以 断言 


sizeof(TestStruct1) 的 结果 为 8。 在 这 个 例子 中 ，sizeof(TestStruct1) 的 结果 为 12。 






































































































































3. 6. 8. 2， 如 何 避 人 免 内 存 对 齐 的 影响 


























那么 ， 能 不 能 既 达 到 提高 性 能 的 目的 ， 又 能 节约 一 点 空间 呢 ? 有 一 点 小 技巧 可 以 使 用 。 
比如 我 们 可 以 将 上 面 的 结构 改 成 : 
struct TestStruct2 


{ 


char cl: 























char c2: 

Short s; 

inti; 
}; 
这 样 一 来 ， 每 个 成 员 都 对 齐 在 其 自然 边界 上 ， 从 而 避免 了 编译 器 自动 对 齐 。 在 这 个 例 
子 中 ，sizeof(TestStruct2) 的 值 为 8。 这 个 技巧 有 一 个 重要 的 作用 ， 尤 其 是 这 个 结构 作为 API 
的 一 部 分 提供 给 第 三 方 开发 使 用 的 时 候 。 第 三 方 开发 者 可 能 将 编译 器 的 默认 对 齐 选 项 改变 ， 
从 而 造成 这 个 结构 在 你 的 发 行 的 DLL 中 使 用 某 种 对 齐 方式 ， 而 在 第 三 方 开发 者 哪里 却 使 用 
另外 一 种 对 齐 方式 。 这 将 会 导致 重大 问题 。 

比如 ，TestStructl 结构 ， 我 们 的 DLL 使 用 默认 对 齐 选项 ， 对 齐 为 

c1 00000000, s 00000002, c2 00000004, i 00000008, 同時 sizeof(TestStruct1) 的 值 为 12。 
而 第 三 方 将 对 齐 选项 关闭 ， 导 致 

c1 00000000, s 00000001, c2 00000003, i 00000004, 同時 sizeof(TestStruct1) 的 值 为 8。 
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除 此 之 外 我 们 还 可 以 利用 #pragma pack O 来 改变 编译 器 的 默认 对 齐 方式 〈 当 然 一 般 编 译 器 


















































也 提供 了 一 些 改变 对 齐 方式 的 选项 ， 这 里 不 讨论 )。 


使 用 指令 #pragma pack (n)， 编 译 器 将 按照 n 个 字 节 对 齐 。 
































按 n 字 节 对 齐 ,但 3 

















使 用 指令 #pragma pack ()， 编 译 器 将 取消 自 定义 字 节 对 齐 方式 。 
在 #pragma pack (n) 利 #pragma pack (0 之 间 的 代码 按 n 个 字 节 对 齐 。 
但 是 ， 成 员 对 齐 有 一 个 重要 的 条 件 , 即 每 个 成 员 按 自己 的 方式 对 齐 . 也 就 是 说 虽然 指定 了 























不 是 所 有 的 成 员 都 是 以 n 字 节 对 齐 。 其 对 齐 的 规则 是 ,每 个 成 员 按 



































的 对 齐 参 数 (通常 是 这 个 类 型 的 大 小 ) 和 指定 对 齐 参数 (这 里 是 n 字 节 ) 中 较 小 的 一 个 对 齐 , 即 : 


min( n, sizeof( item )) 。 并 且 结 构 的 长 度 必须 为 所 用 过 的 所 有 对 齐 参 数 的 整数 倍 , 不 够 部 




















字 节 。 看 如 下 例子 : 











#pragma pack(8) 
struct TestStruct4 
{ 
char a; 
long b; 
}; 
struct TestStructS 
{ 
char c; 
TestStruct4 d; 
long long e; 
}; 
#pragma pack() 


问题 : 


A),sizeof(TestStruct5) = ? 
B), TestStructS 的 c 后 面 空 了 几 个 字 节 接着 是 d? 


TestStruct4 中 ,成 员 a 是 1 字 节 默认 按 1 字 节 对 齐 ,指定 对 齐 参数 为 8, 这 两 个 值 中 取 La 

按 1 字 节 对 齐 ;成 员 b 是 4 个 字 节 ,默认 是 按 4 字 节 对 齐 , 这 时 就 按 4 字 节 对 齐 , 所 以 
sizeof(TestStruct4) 应 该 为 8: 

TestStruct5 中 ,c 和 TestStruct4 中 的 a 一 样 , 按 1 字 节 对 齐 , 而 d 是 个 结构 , 它 是 8 个 字 节 , 它 








按 什 么 对 齐 呢 ? 对 于 
的 一 个 , TestStruct4 的 就 是 4. 所 以 ,成 员 d 1 
字 节 对 齐 , 和 指定 的 一 样 ,所 以 它 对 到 8 字 节 的 边界 J 





























ass 









































结构 来 说 , 它 的 默认 对 齐 方式 就 是 它 的 所 有 成 员 使 用 的 对 齐 参数 中 最 大 






































是 按 4 字 节 对 齐 . 成 员 e 是 8 个 字 节 , 它 是 默认 按 8 
上 ,这 时 ,已 经 使 用 了 12 个 字 节 了 ,所 以 又 添 


加 了 4 个 字 贡 的 空 ,从 第 16 个 字 节 开始 放置 成 员 e. 这 时 ,长 度 为 24, 已 经 可 以 被 8( 成 员 e 按 8 





字 节 对 齐 ) 整 除 .这 样 ,一 共 使 用 了 24 个 字 节 .内 存 布局 如 下 (* 表 示 空 闲 内 存 , 1 表示 使 











单位 为 1byete): 















































a b 


TestStruct4 的 内 存 布局 : 1***,1111, 


C TestStruct4.a TestStruct4.b d 








TestStruct$ 的 内 存 布局 :1***， 14%, 1111, まま まま 。 11111111 


JAT o 


这 里 有 三 点 很 重要 : 























首先 ， 每 个 成 员 分 别 按 自己 的 方式 对 齐 ,并 能 最 小 化 长 度 。 
其 次 , 复杂 类 型 (如 结构 ) 的 默认 对 齐 方 式 是 它 最 长 的 成 员 的 对 齐 方式 ,这 样 在 成 员 是 复杂 
类 型 时 ,可 以 最 小 化 长 度 。 

然后 ， 对 齐 后 的 长 度 必 须 是 成 员 中 最 大 的 对 齐 参 数 的 整数 倍 ,这 样 在 处 理 数组 时 可 以 保 
证 每 一 项 都 边界 对 齐 。 


补充 一 下 ,对 于 数组 ,比如 :char a[3]; 它 的 对 齐 方式 和 分 别 写 3 个 char 是 一 样 的 .也 就 是 说 


















































它 还 是 按 1 个 字 节 对 齐 .如 果 写 : typedef char Array3[3]:Array3 这 种 类 型 的 对 齐 方式 还 是 按 1 
个 字 节 对 齐 ,而 不 是 按 它 的 长 度 。 











但 是 不 论 类 型 是 什么 ,对 齐 的 边界 一 定 是 1,2,4,8,16,32,64.…. 中 的 一 个 。 
另外 ， 注 意 别 的 #pragma pack 的 其 他 用 法 : 
#pragma pack(push) ”// 保 存 当前 对 其 方式 到 packing stack 























#pragma pack(push.n) ”等 效 于 


#pragma pack(push) 





#pragma pack(n) //n=1,2,4,8,16 保存 当前 对 齐 方式 ， 设 置 按 n 字 节 对 齐 
#pragma pack(pop) //packing stack 出 栈 ， 并 将 对 其 方式 设置 为 出 栈 的 对 齐 方 

















3. 7，# 运 算 符 





# 也 是 预 处 理 ? 是 的 ， 你 可 以 这 么 认为 。 那 怎么 用 它 呢 ? 别 乱 ， 先 看 下 面 例子 : 


#define SOR(x) printf("The squareof x is %4d.w",((x)*(x)); 
如 果 这 样 使 用 宏 : 

SQR(8); 

则 输出 为 : 

The square of x is64. 





























SAEN 
注意 





到 没有 ,引号 中 的 字符 x 被 当 作 普通 文本 来 处 理 ， 而 不 是 被 当 作 一 个 可 以 被 蔡 换 的 语言 
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符号 


化 为 

















假如 你 确实 希望 在 字符 串 中 包含 宏 参数 ， 那 我 们 就 可 以 使 用 “#” 它 可 以 把 语言 符号 转 
字符 串 。 上 面 的 例子 改 一 改 : 

#define SQR(x) printf("The square of "#x" is %d.\n", ((x)*(x))); 

HEH: 
SQR(8); 
则 输出 的 是 : 
The square of 8 is64. 

很 简单 吧 ? 相信 你 现在 已 经 明白 # 号 的 使 用 方法 了 。 










































































3.8, HMAK 

















和 # 运 算 符 一 样 ， 检 运算 符 可 以 用 于 宏 函 数 的 蔡 换 部 分 。 这 个 运算 符 把 两 个 语言 符号 组 














合成 单个 语言 符号 。 看 例子 : 
#define XNAME(n) x##n 
如 果 这 样 使 用 宏 : 
XNAME(8) 
则 会 被 展开 成 这 样 : 
X8 
看 明白 了 没 ? 故 就 是 个 粘 合剂 ， 将 前 后 两 部 分 粘 合 起 来 。 
































第 四 章 指针 和 数组 


























几乎 每 次 讲课 讲 到 指针 和 数组 时 ， 我 总 会 反复 不 停 的 问 学 生 : 到 底 什 么 是 指针 ? 什么 
是 数组 ? 他 们 之 间 到 底 是 什么 样 的 关系 。 从 几乎 没 人 能 回答 明白 到 几乎 都 能 回答 明白 ， 需 
HAJR BAAT” 的 痛 。 指針 是 C/C++ 的 精华 ， 如 果 未 能 很 好 地 掌握 指针 ， 那 C/C++ 
也 基本 等 于 没 学 。 可 惜 ， 对 于 刚 毕 业 的 计算 机 系 的 学 生 ， 几 乎 没有 人 真正 完全 掌握 了 指针 
和 数组 、 以 及 内 存 管理 ， 甚 至 有 的 学 生 告 诉 我 说 : 他 们 老师 认为 指针 与 数组 太 难 ， 工 作 又 
少 用 ， 所 以 没有 讲解 。 对 于 这 样 的 学 校 与 老师 ， 我 是 彻底 的 无 语 。 我 没有 资格 去 谴责 或 是 
僵 视 谁 ， 只 是 窃 以 为 ， 这 个 老师 肯 怕 自己 都 未 掌握 指针 。 大 学 里 很 多 老师 并 未 真正 写 过 多 
少 代 码 ， 不 掌握 指针 的 老师 肯定 存在 ， 这 样 的 老师 教 出 来 的 学 生 如 何 能 找到 工作 ? 而 目前 
市 面 上 的 书 对 指针 和 数组 的 区 别 也 是 几乎 避 而 不 谈 ， 这 就 更 加 加 深 了 学 生 掌 握 的 难度 。 我 
平时 上 课 总 是 非常 细致 而 又 小 心 的 向 学 生 讲解 这 些 知识 ， 生 怕 一 不 小 心 就 讲 错 或 是 误导 J 
学 生 。 还 好 ， 至 少 到 目前 为 止 ， 我 教 过 的 学 生 几 乎 都 能 掌握 指针 和 数组 及 内 存 管理 的 要 点 ， 
当然 要 到 能 运用 自如 的 程度 还 远 远 不 够 ， 这 需要 大 量 的 写 代 码 才 能 达到 。 另 外 需要 说 明 的 
是 ， 讲 课时 为 了 让 学 生 深 刻 的 掌握 这 些 知识 ， 我 举 了 很 多 各 式 各 样 的 例子 来 帮助 学 生理 解 。 
所 以 ， 我 也 希望 读者 朋友 能 好 好 体味 这 些 例子 。 











































































































































































































































































































三 个 问题 : 
A)， 什 么 是 指针 ? 
B)， 什 么 是 数组 ? 
C)， 数 组 和 指针 之 间 有 什么 样 的 关系 ? 














4.1, 指針 


4.1.1, 指針 的 内 存 布 局 





先 看 下 面 的 例子 : 

int *p; 

大 家 都 知道 这 里 定义 了 一 个 指针 p。 但 是 p 到 底 是 什么 东西 呢 ? 还 记得 第 一 章 里 说 过 ， 
“任何 一 种 数据 类 型 我 们 都 可 以 把 它 当 一 个 模子 " 吗 ? p， 坚 无 疑问 ， 是 某 个 模子 味 出 来 的 。 
我 们 也 讨论 过 ， 任 何 模子 都 必须 有 其 特定 的 大 小 ， 这 样 才能 用 来 “ 味 味 味 ”。 那 味 出 p 的 这 
个 模子 到 底 是 什么 样子 呢 ? 它 占 多 大 的 空间 呢 ? 现在 用 sizeof 测试 一 下 (32 位 系统 ): sizeof 
(p) 的 值 为 4。 嗯 ， 这 说 明 味 出 p 的 这 个 模子 大 小 为 4 个 byte。 显 然 ， 这 个 模子 不 是 “int”， 
虽然 它 大 小 也 为 4。 既 然 不 是 “int” 那 就 一 定 是 “int*” 了 。 好 ， 那 现在 我 们 可 以 这 么 理解 
这 个 定义 : 


一 个 “int*” 类 型 的 模子 在 内 存 上 味 出 了 4 个 字 节 的 空间 ， 然 后 把 这 个 4 个 字 节 大 小 的 

































































































































































空间 命名 为 p， 同 时 限定 这 4 个 字 节 的 空间 里 面 只 能 存储 某 个 内 存 地 址 ， 即 使 你 存 入 别 的 任 
何 数 据 ， 都 将 被 当 作 地 址 处 理 ， 而 且 这 个 内 存 地 址 开始 的 连续 4 个 字 节 上 只 能 存储 某 个 int 
类 型 的 数据 。 

这 是 一 段 咬 文 嚼 字 的 说 明 ， 我 们 还 是 用 图 来 解析 一 下 : 


指针 示意 图 : 指针 p 指 向 地 址 为 0x0000FF00 的 内 存 


某 内 存 地 址 某 int 类 型 整数 



























































起 始 地 址 为 这 4byte 的 空间 没 


的 起 始 地 4byte 的 空间 ， 名 字 0x0000FF00 ab 
址 未 知 为 p。 名 字 p 与 这 4 个 T. 
字 节 的 空间 一 旦 匹配 访问 
上 就 不 能 被 改变 。p 
本 身 的 起 始 地 址 也 不 
可 改变 











如 上 图 所 示 ， 我 们 把 p 称 为 指针 变量 ,p 里 存储 的 内 存 地 址 处 的 内 存 称 为 p 所 指向 的 内 存 。 
指针 变量 p 里 存储 的 任何 数据 都 将 被 当 作 地 址 来 处 理 。 
我 们 可 以 简单 的 这 么 理解 : 一 个 基本 的 数据 类 型 (包括 结构 体 等 自 定义 类 型 ) 加 上 “*” 
号 就 构成 了 一 个 指针 类 型 的 模子 。 这 个 模子 的 大 小 是 一 定 的 ， 与 “*” 号 前 面 的 数据 类 型 无 
关 。“*” 号 前 面 的 数据 类 型 只 是 说 明 指针 所 指向 的 内 存 里 存储 的 数据 类 型 。 所 以 ， 在 32 位 
系统 下 ， 不 管 什 么 样 的 指针 类 型 ， 其 大 小 都 为 4byte。 可 以 测试 一 下 sizeof (void *)。 






















































































4.1.2,“*” 与 防盗 门 的 钥匙 
































这 里 这 个 “*” 号 怎么 理解 呢 ? 举 个 例子 : 当 你 回 到 家 门口 时 ， 你 想 进 屋 第 一 件 事 就 是 
拿 出 钥匙 来 开锁 。 那 你 想 想 防盗 门 的 锁 蕊 是 不 是 很 像 这 个 “*” 号 ? 你 要 进 屋 必须 要 用 钥匙 ， 
那 你 去 读 写 一 块 内 存 是 不 是 也 要 一 把 钥匙 呢 ? 这 个 “*” 号 就 是 不 是 就 是 我 们 最 好 的 钥匙 ? 
使 用 指针 的 时 候 ， 没 有 它 ， 你 是 不 可 能 读 写 某 块 内 存 的 。 




























































































4.1.3, int *p = NULL 和 *p = NULL 有 什么 区 别 ? 








很 多 初学 者 都 无 法 分 清 这 两 者 之 间 的 区 别 。 我 们 先 看 下 面 的 代码 : 

int *p = NULL: 
这 时 候 我 们 可 以 通过 编译 器 查看 p 的 值 为 0x00000000。 这 人 句 代 码 的 意思 是 : 定义 一 个 指针 
变量 p， 其 指向 的 内 存 里 面 保存 的 是 int 类 型 的 数据 ; 在 定义 变量 p 的 同时 把 p 的 值 设置 为 
0x00000000， 而 不 是 把 *p 的 值 设 置 为 0x00000000。 这 个 过 程 叫 做 初始 化 ， 是 在 编译 的 时 候 
进行 的 。 






































で 
































明白 了 什么 是 初始 化 之 后 ， 再 看 下 面 的 代码 : 
int *p; 
*p = NULL; 
同样 ， 我 们 可 以 在 编译 器 上 调试 这 两 行 代码 。 第 一 行 代码 ， 定 义 了 一 个 指针 变量 p， 其 指向 
的 内 存 里 面 保存 的 是 int 类 型 的 数据 ;但 是 这 时 候 变量 p 本 身 的 值 是 多 少 不 得 而 知 ， 也 就 是 
说 现在 变量 p 保存 的 有 可 能 是 一 个 非法 的 地 址 。 第 二 行 代码 ， 给 部 赋值 为 NULL， 即 给 p 
指向 的 内 存 赋值 为 NULL; 但 是 由 于 p 指向 的 内 存 可 能 是 非法 的 , 所 以 调试 的 时 候 编译 器 可 
能 会 报告 一 个 内 存 访 问 错误 。 这样 的 话 , 我 们 可 以 把 上 面 的 代码 改写 改写 , 使 p 指向 一 块 合 
法 的 内 存 : 
inti = 10; 
int *p = &i; 
*p = NULL; 
在 编译 器 上 调试 一 下 , 我 们 发 现 p 指向 的 内 存 由 原来 的 10 变 为 0 了 ; 而 p 本身 的 值 ， 即 内 
存 地 址 并 没有 改变 。 
经 过 上 面 的 分 析 ， 相 信 你 已 经 明白 它们 之 间 的 区 别 了 。 不 过 这 里 还 有 一 个 问题 需要 注 
意 ， 也 就 是 这 个 NULL。 初 学 者 往往 在 这 里 犯错 误 。 
注意 NULL 就 是 NULL， 它 被 宏 定 义 为 0: 
#defineNULL 0 
很 多 系统 下 除了 有 NULL 外 ,还 有 NULCVisual C++ 6.0 上 提示 说 不 认识 NUL)。NUL 是 ASCII 
码 表 的 第 一 个 字符 ， 表 示 的 是 空 字符 ， 其 ASCII 码 值 为 0。 其 值 虽然 都 为 0， 但 表示 的 意思 
完全 不 一 样 。 同 样 ，NULL 和 0 表示 的 意思 也 完全 不 一 样 。 一 定 不 要 混淆 。 
另外 还 有 初学 者 在 使 用 NULL 的 时 候 误 写成 null 或 Null 等。 这 些 都 是 不 正确 的 ，C 语 
对 大 小 写 十 分 敏感 啊 。 当 然 ， 也 确实 有 系统 也 定义 了 ndl, KARE NULL 没有 区 别 ， 
是 你 千 万 不 用 使 用 null， 这 会 影响 你 代码 的 移植 
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4. 1.4， 如 何 将 数值 存储 到 指定 的 内 存 地 址 





假设 现在 需要 往 内 存 Ox12ff7c 地 址 上 存 入 一 个 整 型 数 0x100。 我 们 怎么 才能 做 到 呢 ? 我 
们 知道 可 以 通过 一 个 指针 向 其 指向 的 内 存 地 址 写 入 数据 ， 那 么 这 里 的 内 存 地 址 Ox12ff7c 其 
本 质 不 就 是 一 个 指针 嘛 。 所 以 我 们 可 以 用 下 面 的 方法 : 

int *p = (int *)0x12ff7c; 

*p = 0x100; 

需要 注意 的 是 将 地 址 Ox12ff7c 赋值 给 指针 变量 p 的 时 候 必须 强制 转换 。 至 于 这 里 为 什 
么 选择 内 存 地 址 0x12ff7c， 而 不 选择 别 的 地 址 ， 比 如 0xff00 等 。 这 仅仅 是 为 了 方便 在 Visual 
C++ 6.0 上 测试 而 已 。 如 果 你 选择 0xfto0， 也 许 在 执行 sp = 0x100; 这 条 语句 的 时 候 ， 编 译 器 
会 报告 一 个 内 存 访问 的 错误 ， 因 为 地 址 0xff00 处 的 内 存 你 可 能 并 没有 权力 去 访问 。 既 然 这 
É, 我 们 怎么 知道 一 个 内 存 地 址 是 可 以 合法 的 被 访问 呢 ? 也 就 是 说 你 怎么 知道 地 址 0x12ff7c 
处 的 内 存 是 可 以 被 访问 的 呢 ? 其 实 这 很 简单 ， 我 们 可 以 先 定义 一 个 变量 i， 比 如 : 

inti = 0; 

变量 i 所 处 的 内 存 肯定 是 可 以 被 访问 的 .然后 在 编译 器 的 watch 窗口 上 观察 &i 的 值 不 就 
知道 其 内 存 地 址 了 么 ? 这 里 我 得 到 的 地 址 是 Ox12ff7c, 仅 此 而 已 (不同 的 编译 器 可 能 每 次 给 
变量 i 分 配 的 内 存 地 址 不 一 样 ， 而 刚好 Visual C++ 6.0 每 次 都 一 样 )。 你 完全 可 以 给 任意 一 个 
可 以 被 合法 访问 的 地 址 赋值 。 得 到 这 个 地 址 后 再 把 “inti = 0;” 这 人 句 代 码 删 除 。 一 切 “ 罪 证 ” 































































































































































































































































































销毁 得 一 干 二 净 ， 简 直 是 做 得 天 衣 无 缝 。 
除了 这 样 就 没有 别 的 办 法 了 吗 ? 未 必 。 我 们 甚至 可 以 直接 这 么 写 代 码 : 
*(int *)0x12ff7c = 0x100; 
这 行 代码 其 实 和 上 面 的 两 行 代码 没有 本 质 的 区 别 。 先 将 地 址 0x12ff7c 强制 转换 ， 告 诉 编译 
器 这 个 地 址 上 将 存储 一 个 int 类 型 的 数据 ， 然后 通过 钥匙 “*” 向 这 块 内 存 写 入 一 个 数据 。 
上 面 讨论 了 这 么 多 ， 其 实 其 表达 形式 并 不 重要 ， 重 要 的 是 这 种 思维 方式 。 也 就 是 说 我 
们 完全 有 办 法 给 指定 的 某 个 内 存 地 址 写 入 数据 的 。 




















































































































4. 1. 5， 编 译 器 的 bug? 





另外 一 个 有 意思 的 现象 ， 在 Visual C++ 6.0 调试 如 下 代码 的 时 候 却 又 发 现 一 个 古怪 的 问 








int *p = (int *)0x12ff7c; 

*p = NULL; 

p = NULL; 
在 执行 完 第 二 条 代码 之 后 , 发 现 p 的 值 变 为 0x00000000 了 。 按照 我 么 上 一 节 的 解释 ， 应 该 p 
的 值 不 变 ， 只 是 p 指向 的 内 存 被 赋值 为 0。 难 道 我 们 讲 错 了 吗 ? 别 急 ， 再 试 试 如 下 代码 ; 

inti= 10; 

int *p = (int *)0x12ff7c; 

*p = NULL; 

p = NULL; 

通过 调试 ， 发 现 这 样子 的 话 ，p 的 值 没 有 变 ， 而 p 指向 的 内 存 的 值 变 为 0 了 。 这 与 我 们 
前 面 讲解 的 完全 一 致 。 当 然 这 里 的 i 的 地 址 刚好 是 Ox12ff7c, 但 这 并 不 能 改变 “*p = NULL;” 
这 行 代码 的 功能 

为 了 再 次 测试 这 个 问题 ， 我 又 调试 了 如 下 代码 : 

inti = 10; 

intj = 100; 

int *p = (int *)0x12ff78; 

*p = NULL: 

p=NULL: 

这 里 0x12ff78 刚好 就 是 变量 j 的 地 址 。 这 样 的 话 一 切 正常 ， 但 是 如 果 把 “intj = 
”这 行 代码 删除 的 话 ， 又 出 现 上 述 的 问题 了 。 测 试 到 这 里 我 还 是 不 甘心 ， 编 译 器 怎么 能 犯 这 
低级 错误 呢 ? 于 是 又 接着 进行 了 如 下 测试 : 


unsigned inti = 10; 




















































































































































































































unsigned int Jj = 100; 
unsigned int *p = (unsigned int *)0x12ff78; 
*p = NULL; 
p = NULL; 
得 到 的 结果 与 上 面 完全 一 样 。 当 然 ， 我 还 是 没有 死心 ， 又 进行 了 如 下 测试 : 
char ch= 10; 
char *p = (char *)0x12ff7c; 
*p = NULL; 
p=NULL; 



























































的 值 并 未 变 成 























这 样子 的 话 ， 完 全 正常 。 但 当 我 删除 掉 第 一 行 代码 后 再 测试 ， 这 里 的 p 
0x00000000， 而 是 变 成 了 0x0012ff00， 同 时 *p 的 值 变 成 了 0。 这 又 是 怎么 回 
否认 为 这 是 编译 器 “良心 发 现 ”， 把 *p 的 值 改写 为 0 了 。 
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事 呢 ? 初学 者 是 








如 果 你 真 这 么 认为 ， 那 就 大 错 特 错 了 。 这 里 的 *p 还 是 地 址 Ox12ff7c 上 的 内 容 吗 ? 显然 


不 是 ， 而 是 地 址 0x0012ff00 上 的 内 容 。 至 于 0x12ff7c 为 什么 变 成 0x0012ff00， 则 是 因为 编 
译 器 认为 这 是 把 NULL 赋值 给 char 类 型 的 内 存 ， 所 以 只 是 把 指针 变量 p 的 低地 址 上 的 一 个 




































































字 节 赋值 为 0。 至 于 为 什么 是 低地 址 ， 请 参看 前 面 讲解 过 大 小 端 模 式 相关 内 容 。 
测试 到 这 里 ， 已 经 基本 可 以 肯定 这 是 Visual C++ 6.0 的 一 个 bug。 所 以 平时 一 定 不 要 迷 


















































信 某 个 编译 器 ， 要 相信 自己 的 判断 。 当 然 ， 后 面 还 会 提 到 一 个 我 认为 的 Vis 
个 bug。 还 有 ， 这 个 小 小 的 例子 ， 你 是 否 可 以 在 多 个 编译 器 上 测试 测试 呢 ? 


























4.1.6， 如 何 达到 手中 无 剑 、 胸 中 也 无 剑 的 地 步 



























































ual C++ 6.0 的 一 


噢 ， 上 面 的 讨论 一 不 小 心 就 这 么 多 了 。 这 里 我 为 什么 要 把 这 个 小 小 的 问题 放 到 这 里 长 








篇 大 论 呢 ? 我 是 想 告诉 读者 : 研究 问题 一 定 要 此 钻研 。 千 万 不 要 小 看 茶 一 个 简单 的 事情 ,， 简 

















单 的 事情 可 能 富 含 着 很 多 秘密 。 经 过 这 样 一 番 深 究 ， 相 信 你 也 有 不 少 收获 。 
是 如 此 ,不 要 小 瞧 任 何 一 件 简 单 的 事情 ， 把 简单 的 事情 做 好 也 是 一 种 伟大 。 
































































































































1L 十 年 的 出 车 ， 技 术 精 到 指 哪 打 哪 的 地 步 。 达 到 这 种 程度 是 需要 花 苦 功夫 的 ， 几 十 年 如 一 
天 天 重复 这 件 看 似 很 简单 的 事情 ， 这 不 是 一 般 人 能 做 到 的 。 同 样 的 ， 在 《天 龙 八 部 》 中 ， 暮 




















平时 学 习 工作 也 
劳模 许 振 超 开 了 
























































峰 血 战 聚 贤 庄 的 时 候 ， 一 套 平 平凡 几 的 太 祖 长 拳 打 得 虎 虎 生 威 ， 在 场 的 英雄 无 不 佩服 至 极 ， 








这 也 是 其 昔 练 的 结果 。 我 们 学 习 工 作 同 样 如 此 ， 要 肯 下 苦 功 夫 钻 研 ， 不 要 怕 

















HIRR RIE 














得 不 深 。 其 实 这 也 就 是 为 什么 同一 个 班 的 学 生 , 水 平 会 相差 非常 大 的 最 关键 
往往 是 那些 舍得 钻研 的 学 生 。 我 平时 上 课 教 学 生 的 绝 不 仅仅 是 知识 点 , 更 多 
们 学 习 和 解决 问题 的 方法 。 有 时 候 这 个 过 程 远 比 结论 要 重要 的 多 。 后面 的 内 









































































































































看 出 来 , 我 非常 注重 过 程 的 分 析 , 只 有 你 真正 明白 了 这 些 思考 问题 、 解 决 问题 的 方法 和 过 程 ， 





> 处。 学 得 好 的 ， 
的 时 候 我 在 教 他 


容 ， 你 也 应 该 能 








你 才能 真正 立 于 不 败 之 地 。 所 有 的 问题 对 你 来 说 都 是 一 个 样 ， 没有 本 质 的 区 别 。 解决 任何 问 
题 的 办 法 都 一 致 ， 那 就 是 把 没 见 过 的 、 不 会 的 问题 想法 设法 转换 成 你 见 过 的 、 你 会 的 问题 ; 
至 于 怎么 去 转换 那 就 要 靠 你 的 苦 学 苦 练 了 。 也 就 是 说 你 要 达到 手中 无 剑 , 胸中 也 无 剑 的 地 步 。 















































当然 这 些 只 是 我 个 人 的 领悟 ， 写 在 这 里 希望 能 与 君 共勉 。 











4.2， 数 组 


4. 2.1， 数 组 的 内 存 布局 





先 看 下 面 的 例子 : 


int a[5]; 


























所 有 人 都 明白 这 里 定义 了 一 个 数组 ， 其 包含 了 5 个 int 型 的 数据 。 我 们 可 以 用 a[0],a[1] 
































等 来 访问 数组 里 面 的 每 一 个 元 素 ， 那 么 这 些 元 素 的 名 字 就 是 a[0],a[1]... 吗 ? 
图 : 

















看 下 面 的 示意 




















= 

EE 
[F —p T O] 
a 作为 右 值 
时 ， 代 表 数 这 20 个 byte 空 间 的 名 字 为 
组 首 元 素 的 a, a[0],a[1] 等 为 a 的 元 素 ， 
首 地 址 ， 而 但 并 非 元 素 的 名 字 。 编 译 
非 数 组 的 首 器 只 给 这 20 个 byte 的 空间 
地 址 (整体 ) 取 了 一 个 名 字 ， 
并 没有 为 其 元 素 取 名 字 。 








数组 示意 图 : 数组 包含 5 个 int 类 型 的 数据 


每 个 元 素 都 是 int 
型 数据 








如 上 图 








所 示 ， 当 我 们 定义 一 个 数组 a 时 , 编译 器 根据 指定 的 元 素 个 数 和 元 素 的 类 型 分 配 确 


デラ 
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大 小 (元 素 类 型 大 小 * 元 素 个 数 ) 的 一 块 内 存 ， 并 把 这 块 内 存 的 名 字 命名 为 a。 名 字 a 一 旦 
与 这 块 内 存 匹 配 就 不 能 被 改变 。a[0],a[]] 等 为 a 的 元 素 ， 但 并 非 元 素 的 名 字 。 数 组 的 每 一 个 














>= 


合 











元 素 都 是 没有 名 字 的 。 那 现在 再 来 
sizeof(a) 的 值 为 sizeof(int)*5，32 位 系统 下 为 20。 


sizeof(a[0]) 的 值 为 sizeof(int)，32 位 系统 下 为 4。 





























第 一 章 讲 解 sizeof 关键 字 时 的 几 个 问题 


sizeof(a[5]) 的 值 在 32 位 系统 下 为 4。 并 没有 出 错 ， 为 什么 呢 ? 我 们 讲 过 sizeof 是 关键 字 








不 是 函数 。 函 数 求 值 是 











在 运行 的 时 候 ， 而 关键 字 sizeof 求 值 是 在 编译 的 时 候 。 


FEI 
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然 并 不 存在 
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a[5] 这 个 元 素 ， 但 是 这 里 也 并 没有 去 真正 访问 a[5], 而 是 仅仅 根据 数组 元 素 的 类 型 来 确定 其 





























值 。 所 以 这 里 使 用 a[5] 并 不 会 出 错 。 





























sizeof(&a[0]) 的 值 在 32 位 系 下 为 4， 这 很 好 理解 。 取 元 素 a[0] 的 首 地 址 。 





sizeof(&a) 的 
C++6.0 上 ， 这 个 值 为 20， 我 认为 是 错误 的 。 


4.2.2， 省 政府 和 市 政 的 区 别 ----&a[0] 和 &a 的 区 別 
这 里 &a[0] 和 &a 到 底 有 什么 

















Xy 



















































































值 在 32 位 系统 下 也 为 4, 这 也 很 好 理解 。 取 数组 a 的 首 地 址 。f 


























是 在 Visual 











E? a[0] 是 一 个 元 素 ，a 是 整个 数组 ， 虽 然 &a[0] 和 &a 



































的 值 一 样 ， 但 其 意义 不 一 样 。 前 者 是 数组 首 元 素 的 首 地 址 ， 而 后 者 是 数组 的 首 地 址 。 举 个 
例子 : 湖南 的 省 政府 在 长 沙 ， 而 长 沙 的 市 政府 也 在 长 沙 。 两 个 政府 都 在 长 沙 ， 但 其 代表 的 
意义 完全 不 同 。 这 里 也 是 同一 个 意思 。 
4.2.3， 数 组 名 a 作为 左 值 和 右 值 的 区 别 

简单 而 言 ， 出 现在 赋值 符 “=” 右 边 的 就 是 右 值 ， 出 现在 赋值 符 “=” 左 边 的 就 是 左 值 。 























比如 ,x=y。 


左 值 : 在 这 个 上 下 文 环境 中 ， 编 译 器 认为 x 的 含义 是 x 所 代表 的 地 址 。 
ij 译 器 知道 ， 在 编译 的 时 候 确定 ， 编 译 器 在 一 个 特定 的 
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这 个 地 址 只 有 
上 ， 我 们 完全 不 必 














区 域 保存 这 个 地 





考虑 这 个 地 址 保存 在 哪里 。 


文 个 上 下 文 环 境 中 ， 
到 运行 时 才 知 道 


“可 修改 的 左 值 ” 
容 一 定 是 可 以 被 修改 的 。 换 句 话 


区 别 ， 下 面 就 讨论 一 下 数组 作为 左 值 


右 值 : 在 这 
个 内 容 是 什么 ， 





mj 


ZN 





C 语言 引入 一 个 术语 





表 的 地 址 上 
既然 已 经 明 


的 内 









































左 值 和 右 值 的 








当 a 作为 右 值 











意思 就 











编译 器 认为 y 的 含义 是 y 所 代表 的 地 址 里 面 的 内 容 。 





是 ， 出 


现在 赋值 符 


这 


左边 的 符号 所 代 












































的 时 候 代表 的 是 什么 意 , 





错误 的 。a 作为 


右 值 





时 














其 意义 与 &a[0] 是 一 样 ， 代 表 的 








Hi 
xÉ 
































的 首 地 址 。 这 是 两 三 





3 事 。 








认为 ， 其 具体 实现 细节 
分 配 一 块 内 存 来 存 其 地 址 ， 这 


a 作为 右 值 ， 我 们 清 
不 能 作为 左 值 ! 这 个 











`. 











L. ss 


但 是 注 ; 
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的 意 上 


ID 


\ 是 a 的 首 元 素 的 首 地 址 , 但 
组 的 某 个 元 素 而 无 法 把 数组 当 一 个 总 体 ; 








， 这 仅仅 是 代表 ， 
S 也 就 是 说 编译 器 并 没有 为 数组 a 
一 点 就 与 指针 有 很 大 的 差别 。 


楚 了 其 含义 ， 


错误 几乎 每 




















那 作 为 左 值 呢 ? 


一 个 学 生 都 犯 过 。 

















当 左 值 。 
RIRA 














通过 


CC 


指针 与 数组 之 间 的 恩 思 


很 多 初学 者 弄 不 清 
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任何 关系 ! 
指针 就 是 指针 ， 





其 实 我 们 完全 可 以 把 a 当 
过 分 别 访问 这 些小 块 来 达到 访 


是 这 个 地 址 开始 的 一 











个 普通 的 变量 来 看 ， 



































指针 








问 整个 变量 a 





AB. $B. 


2 じい 2 じい 2 じい 


指针 和 数组 到 底 有 什么 样 的 关系 。 
只 是 他 们 经 常 穿 着 相似 的 衣服 来 逗 你 玩 罢 了 。 


变量 在 32 位 系统 下 ,永远 占 4 个 byte， 























指针 可 以 指向 





数组 就 是 数组 ， 其 大 小 与 元 素 的 类 
E 何 类 型 的 数据 ， 但 





和 个 数 。 数 组 


可以 存 人 








任何 地 方 ， 但 是 不 是 任何 地 方 你 都 





























既然 它们 2 








针 和 数组 是 一 样 的 。 这 就 与 市 面 上 的 C 语言 的 书 有 关 ， 几 乎 没有 一 本 书 把 这 个 问题 讲 透彻 ， 
讲 明白 了 。 
4.3.1， 以 指针 的 形式 访问 和 以 下 标的 形式 访问 

下 面 我 们 就 详细 讨论 讨论 它们 之 间 似是而非 的 一 些 特点 。 例 如 ， 函 数 内 部 有 如 下 定义 ; 




















WRAEK, A 





A),char *p = “abcdef”; 


B),char a[] = “123456”; 














不 能 存 函 数 。 








为 何 很 多 人 把 数组 




















š, 就 是 我 们 只 能 给 非 只 读 变量 赋值 。 














和 右 值 的 情况 : 




















Hl? 很 多 书 认为 是 数组 的 首 地址 ， 
数组 首 元 素 的 首 地 址 ， 
并 没有 一 个 地 方 





编译 器 会 认为 


只 不 
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RIEN 
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和 指针 混淆 呢 ? 














告诉 你 : 





其 实 这 是 非常 
而 不 是 数组 
的 这 人 么 














(这 只 是 简单 























数组 名 作为 左 值 代表 


块 内 存 是 一 个 总 体 , 我 们 只 能 访问 数 
进行 访问 。 SM 
这 个 变量 内 部 分 为 很 多 小 块 ， 


而 无 法 把 a 





他 们 之 间 没 有 


值 为 某 一 个 内 存 的 地 址 。 
能 通过 这 个 指针 变量 访 


型 和 个 数 有 关 。 定 义 数 组 时 必须 指定 


回 色 。 




















其 元 素 的 类 型 

















t 至 很 多 人 认为 指 
































4. 3.1. 1， 以 指针 的 形式 访问 和 以 下 标的 形式 访问 指针 





例子 A) 定 义 了 一 个 指针 变量 p，p 本 身 在 栈 上 占 4 个 byte, p 里 存储 的 是 一 块 内 存 的 首 
地 址 。 这 块 内 存在 静态 区 ， 其 空间 大 小 为 7 个 byte， 这 块 内 存 也 没有 名 字 。 对 这 块 内 存 的 访 
问 完全 是 匿名 的 访问 。 比 如 现在 需要 读 取 字符 “e”， 我 们 有 两 种 方式 : 

1)， 以 指针 的 形式 : *(p+4)。 先 取出 p 里 存储 的 地 址 值 ， 假 设 为 0x0000FF00， 然 后 加 
上 4 个 字符 的 偏 移 量 ， 得 到 新 的 地 址 0x0000FF04。 然 后 取出 0x0000FF04 地 址 上 的 值 。 


2)， 以 下 标的 形式 : p[4]。 编 译 器 总 是 把 以 下 标的 形式 的 操作 解析 为 以 指针 的 形式 的 操 
作 。p[4] 这 个 操作 会 被 解析 成 : 先 取出 p 里 存储 的 地 址 值 ， 然 后 加 上 中 括号 中 4 个 元 素 的 1 
移 量 ， 计 算出 新 的 地 址 ， 然 后 从 新 的 地 址 中 取出 值 。 也 就 是 说 以 下 标的 形式 访问 在 本 质 上 
与 以 指针 的 形式 访问 没有 区 别 ， 只 是 写法 上 不 同 罢 了 。 
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4. 3.1.2， 以 指针 的 形式 访问 和 以 下 标的 形式 访问 数组 








例子 B) 定 义 了 一 个 数组 a，a 拥有 7 个 char 类 型 的 元 素 ， 其 空间 大 小 为 7。 数 组 a 本 身 
EREM. X a 的 元 素 的 访问 必须 先 根据 数组 的 名 字 a 找到 数组 首 元 素 的 首 地 址 , 然后 根据 
偏 移 量 找到 相应 的 值 。 这 是 一 种 典型 的 “具名 + 匿名 ”访问 。 比 如 现在 需要 读 取 字符 57, 
我 们 有 两 种 方式 : 
1), 以 指针 的 形式 : *(a+4)。a 这 时 候 代 表 的 是 数组 首 元 素 的 首 地 址 , 假设 为 0x0000FF00， 
然后 加 上 4 个 字符 的 偏 移 量 ， 得 到 新 的 地 址 0x0000FF04。 然 后 取出 0x0000FFO4 地 址 上 的 
值 。 









































































































































2)， 以 下 标的 形式 : a[4]。 编 译 器 总 是 把 以 下 标的 形式 的 操作 解析 为 以 指针 的 形式 的 操 
作 。af[4] 这 个 操作 会 被 解析 成 : a 作为 数组 首 元 素 的 首 地 址 ， 然 后 加 上 中 括号 中 4 个 元 素 的 
偏 移 量 ， 计 算出 新 的 地 址 ， 然 后 从 新 的 地 址 中 取出 值 。 
由 上 面 的 分 析 ， 我 们 可 以 看 到 ， 指 针 和 数组 根本 就 是 两 个 完全 不 一 样 的 东西 。 只 是 它们 
都 可 以 “以 指针 形式 ”或 “以 下 标 形 式 ” 进 行 访问 。 一 个 是 完全 的 匿名 访问 ， 一 个 是 典型 
的 具名 + 匿名 访问 。 一 定 要 注意 的 是 这 个 “以 XXX 的 形式 的 访问 ”这 种 表达 方式 。 

另外 一 个 需要 强调 的 是 : 上 面 所 说 的 偏 移 量 4 代表 的 是 4 个 元 素 ， 而 不 是 4 个 byte。 只 
不 过 这 里 刚好 是 char 类 型 数据 1 个 字符 的 大 小 就 为 1 个 byte。 记 住 这 个 偏 移 量 的 单位 是 元 
素 的 个 数 而 不 是 byte 数 ， 在 计算 新 地 址 时 干 万 别 弄 错 了 。 




























































































































































































4.3.2, a 和 &a 的 区 列 

















通过 上 面 的 分 析 ， 相 信 你 已 经 明白 数组 和 指针 的 访问 方式 了 ， 下 面 再 看 这 个 例子 : 





























main() 

( 
int a[5]={1,2,3,4,5}; 
int *ptr=(int *)(&a+1); 
printf("%d,%d",* (a+1),* (ptr-1)); 























打印 出 来 的 值 为 多 少 呢 ?这 里 主要 是 考查 关于 指针 加 减 操作 的 理解 。 
对 指针 进行 加 1 操作, 得 到 的 是 下 一 个 元 素 的 地 址 , 而 不 是 原 有 地 址 值 直 接 加 1。 所 以 ， 











一 个 类 型 为 T 的 指针 的 移动 ， 以 sizeof(T) 为 移动 单位 。 
ptr 是 一 个 int 型 的 指针 。 
&a+ 1: 取 数 组 a 的 首 地 址 ， 该 地 址 的 值 加 上 sizeof(a) 的 值 ， 即 wa+ S*sizeofGnt), tB 
就 是 下 一 个 数组 的 首 地 址 ， 显 然 当前 指针 已 经 越过 了 数组 的 界限 。 


维 数组 ， 数 组 中 有 5 个 元 素 ; 






























































因此 ， 对 上 题 来 说 ， td 十 二 个 一 


(int*)(&a+1): 则 是 把 上 一 步 计算 出 来 的 地 址 ， 强 制 转换 为 int* 类 型 ， 赋 值 给 ptr。 




















*(a+1): a,&a 的 值 是 一 样 的 ， 但 意思 不 一 样 ，a 是 数组 首 元 素 的 首 地 址 ， 也 就 是 a[0] 的 
首 地 址 ，&a 是 数组 的 首 地 址 ，a+l 是 数组 下 一 元 素 的 首 地 址 ， 即 a[1] 的 首 地 址 ,&a+l 是 下 一 











个 数组 的 首 地 址 。 所 以 输出 2 
*(ptr-1): 因为 ptr 是 指向 a[5]， 并 
前 出 5. 
































这 些 分 析 我 相信 大 家 都 能 理解 ， 但 是 在 授课 时 ， 
































ptr 是 int* 类 型 ， 所 以 *(ptr-1) 是 指向 a[4] ， 


学 生 向 我 提出 了 如 下 问题 : 


TE Visual C++6.0 的 Watch 窗口 中 &a+tl 的 值 怎么 会 是 (x0012ff6d (0x0012ff6c+1) W? 


Bx00812Fff6C 
&a 
&a+1 Bx00812Ff6d """ 


0x0012ff70 


a+1 
(コキ 1 ) 
*(ptr-1) 











上 图 是 在 Visual C++6.0 调试 本 函数 时 的 截图 。 











Bx HB12FF6C "J 





a 在 这 里 代表 是 的 数组 首 元 素 的 地 址 即 af0] 的 首 地 址 ， 其 值 为 Ox0012ff6c。 
&a 代表 的 是 数组 的 首 地 址 ， 其 值 为 0x0012ff6c。 











a+1 的 值 是 Ox0012ff6c+1*sizeof Gint), 等 3 





F Ox0012ff70。 


问题 就 是 &at1 的 值 怎么 会 是 (x0012ff6d (Ox0012ffec+1) WE? 





按照 我 们 上 


























而 的 分 析 应 该 为 0x0012ff6c+5*sizeof Gnt) KREE 





E 解 。 当 你 把 &a+1 




















中 观察 其 








放 到 Watch 窗 








它 解析 为 &a 的 值 然后 加 上 lbyte。 而 a+1 的 解析 就 了 


























值 时 ， 表 达 式 &atl 已 经 脱离 其 上 下 文 环境 ， 编 译 器 就 很 简单 的 把 
E 确 ， 我 认为 这 是 Visual C++6.0 的 一 个 





bug。 既 然 如 此 , 我 们 怎么 证 明证 明 &a+tl 的 值 确实 为 0x0012ff6c+5*sizeof (int) 呢 ? 很 好 办 ， 





用 printf 函数 打印 出 来 。 这 就 是 我 在 本 书 前 言 里 所 说 的 ， 
才能 解决 问题 。 你 可 以 试 试用 printf("%x",&a+1); 打 印 其 值 ， 看 
的 是 printf("9%d",&a+1); 打 印 ， 那 你 必须 在 十 进 




















(int)。 注 意 如 果 你 
一 下 ， 不 要 多 枉 了 编译 器 。 





















































的 时 候 我 们 确实 需要 printf 函数 


是 否 为 Ox0012ff6c+S*sizeof 























由 和 十 六 进 制 之 间 换 算 





另外 我 要 强调 一 点 : 不 到 非 不 得 已 ， 尽 量 别 使 用 printf 函数 ， 它 会 使 你 养 成 只 看 结果 不 
问 为 什么 的 习惯 。 比 如 这 个 列子 ，*(at+1) 和 *(ptr-1) 的 值 完全 可 以 通过 Watch 窗口 来 查看 。 





























平时 初学 者 很 喜欢 


如 果 发 现 值 是 正确 的 就 欢天喜地 。 这 个 时 候 往往 认为 



































“printf("96d,%d",*(a+1),*(ptr-1));” 这 类 的 表达 式 来 直接 打印 出 值 ， 
自己 的 代码 没有 问题 ， 


根本 就 不 去 查 









































看 其 变量 的 值 ， 更 别 说 是 内 存 和 寄存 器 的 值 了 。 更 有 甚 者 ，printf 函数 打印 出 来 的 值 不 正确 ， 
就 措 手 无 策 ， 举 手 问 “老师 ， 我 这 里 为 什么 不 对 啊 ? ”。 长 此 以 往 就 养 成 了 很 不 好 的 习惯 ， 
只 看 结果 ， 不 重 调试 。 这 就 是 为 什么 同样 的 几 年 经 验 ， 有 的 人 水 平 很 高 ， 而 有 的 人 水 平 却 
很 低 。 其 根本 原因 就 在 于 此 ， 往 往 被 一 些 表面 现象 所 迷惑 。printf 函数 打印 出 来 的 值 是 对 的 
就 能 说 明 你 的 代码 一 定 没 问题 吗 ? 我 看 未 必 。 曾 经 一 个 学 生 ， 我 让 其 实现 直接 插入 排序 算 
法 。 很 快 他 把 函数 写 完了 ， 把 值 用 printf 函数 打印 出 来 给 我 看 。 我 看 其 代码 却 发 现 他 使 用 的 
法 本 质 上 其 实 是 冒 泡 排序 ， 只 是 写 得 像 直接 插入 排序 罢了 。 等 等 这 种 情况 数 都 数 不 过 来 ， 
往往 犯 了 错误 还 以 为 自己 是 对 的 。 所 以 我 平时 上 课 之 前 往往 会 强调 ， 不 到 非 不 得 已 ， 不 允 
许 使 用 printf 函数 ， 而 要 自己 去 查看 变量 和 内 存 的 值 。 学 生 的 这 种 不 好 的 习惯 也 与 目前 市 面 
上 的 教材 、 参 考 书 有 关 ， 这 些 书 甚至 花 大 篇 幅 来 介绍 scanf 和 printf 这 类 的 函数 ， 却 几乎 不 
解 调试 技术 。 甚 至 有 的 书 还 在 讲 TruboC 2.0 之 类 的 调试 器 ! 如 此 教材 教 出 来 的 学 生 质量 
可 想 面 知 。 
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4. 3.3， 指 针 和 数组 的 定义 与 声明 


4. 3. 3. 1， 定 义 为 数组 ， 声 明 为 指针 


文件 1 中 定义 如 下 : 
char a[ 100]; 
文件 2 中 声明 如 下 (关于 extern 的 用 法 ， 以 及 定义 和 声明 的 区 别 ， 请 复习 第 一 章 ); 


extern char *a; 

这 里 ， 文 件 1 中 定义 了 数组 a， 文 件 2 中 声明 它 为 指针 。 这 有 什么 问题 吗 ? 平时 不 是 总 说 数 
组 与 指针 相似 ， 甚 至 可 以 通用 吗 ? 但 是 ， 很 不 垃 ， 这 是 错误 的 。 通 过 上 面 的 分 析 我 们 也 能 
明白 一 些 ， 但 是 “革命 尚未 成 功 ， 同 志 仍 需 努力 ”。 你 或 许 还 记得 我 上 面 说 过 的 话 : 数组 就 
是 数组 ， 指 针 就 是 指针 ， 它 们 是 完全 不 同 的 两 码 事 ! 他 们 之 间 没 有 任何 关系 ， 只 是 经 常 穿 
着 相似 的 衣服 来 迷惑 你 办 了 。 下 面 就 来 分 析 分 析 这 个 问题 : 

在 第 一 章 的 开始 ， 我 就 强调 了 定义 和 声明 之 间 的 区 别 ， 定 义 分 配 的 内 存 ， 而 声明 没有 。 
定义 只 能 出 现 一 次 ,而 声明 可 以 出 现 多 次 。 这 里 extern 告诉 编译 器 a 这 个 名 字 已 经 在 别 的 文 
件 中 被 定义 了 ,下 面 的 代码 使 用 的 名 字 a 是 别 的 文件 定义 的 。 再 回顾 到 前 面 对 于 左 值 和 右 值 
的 讨论 ， 我 们 知道 如 果 编 译 器 需要 某 个 地 址 〈 可 能 还 需要 加 上 偏 移 量 ) 来 执行 某 种 操作 的 
话 ， 它 就 可 以 直接 通过 开锁 动作 (使 用 “*” 这 把 钥匙 ) 来 读 或 者 写 这 个 地 址 上 的 内 存 ， 并 不 
需要 先 去 找到 储存 这 个 地 址 的 地 方 。 相 反 ， 对 于 指针 而 言 ， 必 须 先 去 找到 储存 这 个 地 址 的 
地 方 ， 取 出 这 个 地 址 值 然 后 对 这 个 地 址 进行 开锁 《使 用 “*” 这 把 钥匙 )。 如 下 图 : 























































































































































































































char al] = “abcdefg”; 
在 定义 数组 a 的 时 候 编 译 器 在 某 个 地 方 保存 了 a 的 


首 元 素 的 首 地 址 0x0000FF00。 
0x0000FF00 
+1*sizeof(char) +i*sizeof(char) 
要 取 a[i 的 内 容 分 为 两 步 : 


1， 计 算 a 自 的 地 址 : 0x0000FF00+i*sizeof(char)。 
2, 取 0x0000FF00+i*sizeof(char) 地 坪 上 的 内 容 














这 就 是 为 什么 extern char a[] 与 extern char a[100] 等 价 的 原因 。 因 为 这 只 是 声明 ， 不 分 配 
空间 , 所 以 编译 器 无 需 知道 这 个 数组 有 多 少 个 元 素 。 这 两 个 声明 都 告诉 编译 器 a 是 在 别 的 文 
件 中 被 定义 的 一 个 数组 ，a 同时 代表 着 数组 a 的 首 元 素 的 首 地 址 ， 也 就 是 这 块 内 存 的 起 始 地 
址 。 数 组 内 地 任何 元 素 的 的 地 址 都 只 需要 知道 这 个 地 址 就 可 以 计算 出 来 。 


但 是 ， 当 你 声明 为 extern char *a 时 ， 编 译 器 理所当然 的 认为 a 是 一 个 指针 变量 ， 在 32 位 系 
统 下 ， 占 4 个 byte。 这 4 个 byte 里 保存 了 一 个 地 址 ， 这 个 地 址 上 存 的 是 字符 类 型 数据 。 虽 
然 在 文件 1 中 ,编译 器 知道 a 是 一 个 数组 , 但 是 在 文件 2 中 ,编译 器 并 不 知道 这 点 。 大 多 数 
编译 器 是 按 文件 分 别 编译 的 ,编译 器 只 按照 本 文件 中 声明 的 类 型 来 处 理 。 所以, 虽然 a 实际 
大 小 为 100 條 byte, 但 是 在 文件 2 中 ， 编 译 器 认为 a 只 占 4 个 byte。 


我 们 说 过 ， 编 译 器 会 把 存在 指针 变量 中 的 任何 数据 当 作 地 址 来 处 理 。 所 以 ， 如 果 需 要 
访问 这 些 字符 类 型 数据 ， 我 们 必须 先 从 指针 变量 a 中 取出 其 保存 的 地 址 。 如 下 图 : 





































































































































































































































































































extern char *a; 
编译 器 认为 a 是 一 个 指针 变量 ， 占 4 个 byte。 假 设 
原 数 组 a 中 保持 了 100 个 字符 A、B、C、D… 等 的 
ASCIIA 码 值 ， 但 是 在 这 里 ， 编 译 器 只 能 看 到 前 4 
个 byte 的 空间 。 


le le le [sa 
0x0000FF00 这 个 地 


址 ， 并 没有 用 到 0x828384 

1， 编 译 器 按 int 类 型 的 取 值 方法 一 次 性 取出 前 4 个 byte 

的 值 ， 得 到 10000001100000101000001110000100， 转 

换 十 六 进 制 : 0x828384。 (这 里 先 不 考虑 大 小 端 存储 

模式 ) 

2， 地 址 0x828384 上 的 内 容 ， 按 照 char 类 型 读 写 。 但 

是 地 址 0x828384 可 能 并 非 是 个 有 效 的 地 址 ， 退 一 步 ， 

即使 这 是 个 有 效 的 地 址 ， 那 也 不 是 我 们 想 要 的 。 





4. 3. 3. 2， 定 义 为 指针 ， 声 明 为 数组 

















显然 ,按照 上 面 的 分 析 , 我 们 把 文件 1 中 定义 的 数组 在 文件 2 中 声明 为 指针 会 发 生 错误 。 
同样 的 ， 如 果 在 文件 1 中 定义 为 指针 ， 而 在 文件 中 声明 为 数组 也 会 发 生 错误 : 


文件 1 























char *p = “abcdefg”; 

文件 2 

extern char p[]; 
在 文件 1 中 , 编译 器 分 配 4 个 byte 空间 , 并 命名 为 p。 同 时 p 里 保存 了 字符 串 常量 “abcdefg” 
的 首 字符 的 首 地 址 。 这 个 字符 串 常量 本 身 保 存在 内 存 的 静态 区 ， 其 内 容 不 可 更 改 。 在 文件 2 
中 ， 编 译 器 认为 p 是 一 个 数组 ， 其 大 小 为 4 个 byte， 数 组 内 保存 的 是 char 类 型 的 数据 。 在 
文件 2 中 使用 p 的 过 程 如 下 图 : 































































































指针 p 内 保存 的 是 字符 串 常量 的 地 址 ， 假 设 为 
0x0000FF00。 这 里 先 不 考虑 大 小 端 存储 模 
式 ) 


[=e] 
p 本 身 的 地 址 这 里 


并 没有 用 到 编译 器 把 指针 变量 p 当 作 一 个 包含 4 个 char 类 型 
数据 的 数组 来 使 用 ， 按 char 类 型 取出 p[0]、 
p[1]、p[2]、p[3] 的 値 0x00、0x00、OxEF、 
0x00。 但 这 并 非 我 们 所 要 的 某 块 内 存 的 地 址 。 
如 果 给 p 上 赋值 则 会 把 原来 p 中 保持 的 真正 地 址 
覆盖 ， 导 致 再 也 无 法 找到 其 原来 指向 的 内 存 。 


4. 3. 4， 指 针 和 数组 的 对 比 








通过 上 面 的 分 析 ， 相 信 你 已 经 知道 数组 与 指针 的 的 确 确 是 两 码 事 了 。 他 们 之 间 是 不 可 
以 混淆 的 ， 但 是 我 们 可 以 “以 XXXX 的 形式 ”访问 数组 的 元 素 或 指针 指向 的 内 容 。 以 后 一 
定 要 确认 你 的 代码 在 一 个 地 方 定义 为 指针 ， 在 别 的 地 方 也 只 能 声明 为 指针 ;在 一 个 的 地 方 
定义 为 数组 ， 在 别 的 地 方 也 只 能 声明 为 数组 。 切 记 不 可 混淆 。 下 面 再 用 一 个 表 来 总 结 一 下 
指针 和 数组 的 特性 : 

















































































































指针 数组 














保存 数据 的 地 址 ， 任 何 存 入 指针 变量 p 的 数 | 保存 数据 ， 数 组 名 a 代表 的 是 数组 首 元 素 的 
据 都 会 被 当 作 地 址 来 处 理 。p 本 身 的 地 址 由 | 首 地 址 而 不 是 数组 的 首 地 址 。&a 才 是 整个 数 
编译 器 另外 存储 ， 存 储 在 哪里 ， 我 们 并 不 知 | 组 的 首 地 址 。a 本 身 的 地 址 由 编译 器 另外 存 





























































































































いさ と 


1E 。 























储 ， 存 储 在 哪里 ， 我 们 并 不 知道 。 























间接 访问 数据 , 首先 取得 指针 变量 p 的 内 容 ， 
把 它 作 为 地 址 ， 然 后 从 这 个 地 址 提取 数据 或 





直接 访问 数据 ,数组 名 a 是 整个 数组 的 名 字 ， 
数组 内 每 个 元 素 并 没有 名 字 。 只 能 通过 “ 具 





















































向 这 个 地 址 写 入 数据 。 指 针 可 以 以 指针 的 
式 访问 *(p+i); 也 可 以 以 下 标的 形式 访问 plil- 
但 其 本 质 都 是 先 取 p 的 内 容 然后 加 上 


SSN 
























































名 + 匿名 ”的 方式 来 访问 其 某 个 元 素 , 不能 
数组 当 一 个 整体 来 进行 读 写 操作 。 数 组 可 以 
以 指针 的 形式 访问 *(a+iD; 也 可 以 以 下 标的 形 






















































































i*sizeof( 类 型 ) 个 byte 作为 数据 的 真正 地 址 。 式 访 问 ail。 但 其 本 原 都 赴 a 所 代表 的 数组 首 
元 素 的 首 地 址 加 上 ixsizeof( 类 型 ) 个 byte 作为 
数据 的 真正 地 址 。 

通常 用 于 动态 数据 结构 通常 用 于 存储 固定 数目 且 数 据 类 型 相同 的 元 
素 。 

相关 的 函数 为 malloc 和 free。 隐 式 分 配 和 删除 

通常 指向 匿名 数 据 ( 当 然 也 可 指向 具 名 数 据 ) | 自身 即 为 数组 名 

















4. 4， 指 针 数 组 和 数组 指针 


4. 4.1， 指 针 数 组 和 数组 指针 的 内 存 布 局 


初学 者 总 是 分 不 出 指针 数组 与 数组 指针 的 
个 数组 ， 数 组 的 元 素 都 是 指针 ， 数 组 占 多 少 


























指针 数组 : 首先 它 是 
决定 。 它 是 “储存 指针 的 数组 ”的 简称 。 


数组 指针 : 首先 它 是 一 个 指针 ， 它 指向 一 





> 














区 别 。 其 实 很 好 





Hg: 




















个 字 节 由 数组 本 身 





ブ 、 と テオ キモ 


个 数组 。 在 32 位 系统 下 永远 是 占 4 个 字 节 ， 











至 于 它 指向 的 数组 占 多 少 字 节 ， 不 知道 。 




















下 面 到 底 哪个 是 数组 指针 ， 虽 
A), int *pl[10]; 


B), int (*p2)[10]; 





每 次 上 课 问 这 个 问题 ， 总 有 型 不 清楚 的 。 
“ 吕 ” 的 优先 级 比 “*” 要 高 。pl 先 与 “ 口 ”结合 ， 构 成 一 








已 下 


是 “指向 数组 的 指针 ”的 简称 。 


P 个 是 指针 数组 呢 : 





要 明白 一 个 符号 之 间 的 优先 级 问题 。 
个 数组 的 定义 ， 数 组 名 为 pl，int * 





这 里 需 

















修饰 的 是 数组 的 内 容 ， 即 数组 的 每 个 元 素 。 





指向 int 类 型 数据 的 指针 ， 即 指针 数组 。 至 于 p2 就 更 好 


“ 口 ”高 ,“*” 号 和 p2 构成 一 个 指针 的 定义 








针 ， 


理解: 





即 数组 的 每 个 元 素 。 数 组 在 这 里 并 没有 名 字 ， 
它 指向 一 个 包含 10 int 类 型 数据 的 数组 ， 即 数组 指针 。 我 们 可 以 借 



































那 现 在 我 们 清楚 ， 这 是 一 个 数组 ， 其 包含 10 个 
里 解 了 ， 在 这 里 “()” 的 优先 级 比 























， 指 针 变 量 名 为 p2, int 修饰 的 是 数组 的 内 容 ， 
是 个 匿名 数组 。 那 现在 我 们 一 个 指 








N= = =l 
清楚 p2 是 


助 下 面 的 图 加 深 























pl 
int *p1[10]; 











PpI 为 数组 名 


int (“p2)l10]; 





0x0000FF00 p2 为 指针 变量 名 


>= a — PENTE 人 -— 二 一 人 
| int int int int int | int int int int int | 
Ox0000FEOO 


4.4.2, int (*) [10] p2—— 也 许 应 该 这 么 定义 数组 指针 


这 里 有 个 有 意思 的 话题 值得 探讨 一 下 : 平时 我 们 定义 指针 不 都 是 在 数据 类 型 后 面 加 上 
指针 变量 名 么 ? 这 个 指针 p2 的 定义 怎么 不 是 按照 这 个 语法 来 定义 的 呢 ? 也 许 我 们 应 该 这 样 
来 定义 p2: 

int [10] p2; 


int (*)[10] 是 指针 类 型 ，p2 是 指针 变量 。 这 样 看 起 来 的 确 不 错 ， 不 过 就 是 样子 有 些 别 
扭 。 其 实数 组 指针 的 原型 确实 就 是 这 样子 的 ， 只 不 过 为 了 方便 与 好 看 把 指针 变量 p2 前 移 了 
而 已 。 你 私下 完全 可 以 这 么 理解 这 点 。 虽 然 编译 器 不 这 么 想 。^^ 






















































































































































































4.4.3， 再 论 a 和 &a 之 间 的 区 别 








既然 这 样 ， 那 问题 就 来 了 。 前 面 我 们 讲 过 a Mea 之 间 的 区 别 ， 现 在 再 来 看 看 下 面 的 代 
码 : 




















int main() 

{ 
char a[5]={'A',B','C','D'}; 
char (*p3)[5] = &a; 
char (*p4)[5] = a; 


return 0; 








上 面 对 p3 和 p4 的 使 用 ， 哪 个 正确 呢 ? p3+1 的 值 会 是 什么 ? ph 的 值 又 会 是 什么 ? 
































训 无 疑问 ，p3 和 p4 都 是 数组 指针 ， 指 向 的 是 整个 数组 。&a 是 整个 数组 的 首 地 址 ，a 
是 数组 首 元 素 的 首 地 址 ， 其 值 相同 但 意义 不 同 。 在 C 语言 里 ， 赋 值 符 号 “=” 号 两 边 的 数据 
类 型 必须 是 相同 的 ， 如 果 不 同 需要 显示 或 隐 式 的 类 型 转换 。p3 这 个 定义 的 “=” 号 两 边 的 数 
据 类 型 完全 一 致 ， 而 p4 这 个 定义 的 “=” 号 两 边 的 数据 类 型 就 不 一 致 了 。 左 边 的 类 型 是 指 
向 整个 数组 的 指针 ， 右 边 的 数据 类 型 是 指向 单个 字符 的 指针 。 在 Visual C++6.0 上 给 出 如 下 
警告 : warning C4047: 'initializing' : char (*)[5]' differs in levels of indirection from char *'。 还 好 ， 
这 里 虽然 给 出 了 警告 , 但 由 于 &a 和 a 的 值 一 样 ， koa a 
所 以 运行 并 没有 什么 问题 。 不 过 我 仍然 警告 你 别 这 么 用 。 

既然 现在 清楚 了 p3 和 p4 都 是 指向 整个 数组 的 ， 那 p3+1 和 p4+1 的 值 就 很 好 理解 了 。 
但 是 如 果 修 改 一 下 代码 ， 会 有 什么 问题 ? p3+1 和 p4+1 的 值 又 是 多 少 呢 ? 

















































































































































































































int main() 
{ 
char a[5]={'A','B','C','D'}; 
char (*p3)[3] = &a 
char (*p4)[3] = a; 
return 0; 


} 
至 还 可 以 把 代码 再 修改 : 


int main() 


{ 




















char a[5]={'A','B','C','D'}; 
char (*p3)[10] = 

char (*p4)[10] = a; 
return 0; 


} 
这 个 时 候 又 会 有 什么 样 的 问题 ? p3+1 和 p4+1 的 值 又 是 多 少 ? 


上 述 几 个 问题 ， 希 望 读者 能 仔细 考虑 考虑 。 


























4.4.4， 地 址 的 强制 转换 





先 看 下 面 这 个 例子 : 
Struct Test 


( 

















Int Num; 
char *pcName; 


short sDate: 
char cha[2]; 
short sBal4]: 
PP: 
假设 p 的 值 为 Ox100000。 如 下 表 表 达 式 的 值 分 别 为 多 少 ? 
D+Ox1=Ox_ ? 
(unsigned long)p +0x1 = 0x__? 
(unsigned int*)p+Ox1=Ox_ ? 
我 相信 会 有 很 多 人 台 没 看 明白 这 个 问题 是 什么 意思 。 其 实 我 们 再 仔细 看 看 ， 这 个 知识 点 
似 兽 相识。 一 个 指针 变量 与 一 个 整数 相 加 减 ， 到 底 该 怎么 解析 呢 ? 
还 记得 前 面 我 们 的 表达 式 “a+1” 与 “&a+1” 之 间 的 区 别 吗 ? 其 实 这 里 也 一 样 。 指 针 变 
量 与 一 个 整数 相 加 减 并 不 是 用 指针 变量 里 的 地 址 直接 加 减 这 个 整数 。 这 个 整数 的 单位 不 是 
byte 而 是 元 素 的 个 数 。 所 以 : 
p +0x1 的 值 为 0x100000+sizof (Test) *0x1。 至 于 此 结构 体 的 大 小 为 20byte， 前 面 的 章 
节 已 经 详细 讲解 过 。 所 以 p+0x1 的 值 为 : 0x100014。 
(unsigned long)p + Ox1 的 值 呢 ? 这 里 涉及 到 强制 转换 ， 将 指针 变量 p 保存 的 值 强制 转换 
成 无 符号 的 长 整 型 数 。 任 何 数值 一 旦 被 强制 转换 ， 其 类 型 就 改变 了 。 所 以 这 个 表达 式 其实 就 
是 一 个 无 符号 的 长 整 型 数 加 上 男 一 个 整数 。 所 以 其 值 为 : 0x100001。 
(unsigned int*)p + 0x1 的 值 呢 ? 这 里 的 p 被 强制 转换 成 一 个 指向 无 符号 整 型 的 指针 。 所 
以 其 值 为 : 0x100000+sizof (unsigned int) *Ox1, 等 耳 0x100004。 
上 上 面 这 个 问题 似乎 还 没 喻 技术 含量 ， 下 面 就 来 个 有 技术 含量 节 
在 x86 系统 下 ， 其 值 为 多 少 ? 
int main() 


{ 












































































































































































































































int a[4]={ 1,2,3,4}; 
int *ptrl=(int *)(&a+1); 
int *ptr2=(int *)((int)a+1); 


printf("%x,%x",ptrl[-1],*ptr2); 


return 0; 

} 
这 是 我 讲课 时 一 个 学 生 问 我 的 题 , 他 在 网 上 看 到 的 , 据说 难 倒 了 n 个 人 。 我 看 题 之 后 告诉 他 ， 
这 些 人 肯定 不 懂 汇 编 , 一 个 懂 汇编 的 人 , 这 种 题 实在 是 小 case。 下 面 就 来 分 析 分 析 这 个 问题 : 
民 据 上 面 的 讲解 ，&a+1l 与 a+1 的 区 别 已 经 清楚 。 

ptrl: 将 &a+l 的 值 强制 转换 成 int* 类 型 赋值 给 int* 类 型 的 变量 ptr, prl 育 定 指 到 数 
组 a 的 下 一 个 int 类 型 数据 了 。ptrl[-1] 被 解析 成 *(ptr1-1)， 即 ptrl 往 后 退 4 个 byte。 所 以 其 
值 为 0x4。 

ptr2: 按照 上 面 的 讲解 ，(intjat1l 的 值 是 元 素 a[0] 的 第 二 个 字 节 的 地 址 。 然 后 把 这 个 地 址 
强制 转换 成 int* 类 型 的 值 赋 给 ptr2, 也 就 是 说 *ptr2 的 值 应 该 为 元 素 a[0] 的 第 二 个 字 节 开始 的 
连续 4 个 byte 的 内 容 。 
其 内 存 布局 如 下 图 : 









































































































































































































































int *ptrl=(int *)(&a+l); 


int *ptr2=(int *)((int)a+1); 数组 a 








PN, : 
a[0] | all] | a[2] al3] N 
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Wa 
连续 4 个 byte 组 成 
一 个 int 类 型 数 


pt2 ptrl 






































好 , 问题 就 来 了 ,这 连续 4 个 byte 里 到 底 存 了 什么 东西 呢 ? 也 就 是 说 元 素 a[O],a[1] 里 面 
的 值 到 底 怎 么 存储 的 。 这 就 涉及 到 系统 的 大 小 端 模式 了 ， 如 果 懂 汇编 的 话 ， 这 根本 就 不 是 问 
题 。 既 然 不 知道 当前 系统 是 什么 模式 ， 那 就 得 想 办 法 测试 。 大 小 端 模式 与 测试 的 方法 在 第 一 
章 讲解 union 关键 字 时 已 经 详细 讨论 过 了 ， 请 翻 到 彼 处 参看 ， 这 里 就 不 再 详 述 。 我 们 可 以 用 
下 面 这 个 函数 来 测试 当前 系统 的 模式 。 

int checkSystem( ) 

















































































































( 
union check 
( 
int i; 
char ch 
) c; 
ci=l; 
return (c.ch ==1 ): 
} 


如 果 当 前 系统 为 大 端 模式 这 个 函数 返回 0， 如 果 为 小 端 模式 ， 函 数 返 回 1 
也 就 是 说 如 果 此 函数 的 返回 值 为 1 的 话 ，*ptr2 的 值 为 Ox2000000。 
如 果 此 函数 的 返回 值 为 0 的 话 ，*ptr2 的 值 为 Ox100。 





























4.5， 多 维 数组 与 多 级 指针 





多 维 数组 与 多 级 指针 也 是 初学 者 感觉 迷糊 的 一 个 地 方 。 超 过 二 维 的 数组 和 超过 二 级 的 
指针 其 实 并 不 多 用 。 如 果 能 卉 明白 二 维 数 组 与 二 级 指针 ， 那 二 维 以 上 的 也 不 是 什么 问题 了 。 
所 以 本 节 重 点 讨论 二 维 数 组 与 二 级 指针 。 
































4. 5.1， 二 维 数组 


4. 5. 1.1， 假 想 中 的 二 维 数组 布局 














我 们 前 面 讨论 过 ， 数 组 里 面 可 以 存 任何 数据 ， 除 了 函数 。 下 面 就 详细 讨论 讨论 数组 里 
面 存 数组 的 情况 .Excel K, 我 相信 大 家 都 见 过 。 我 们 平时 就 可 以 把 二 维 数组 假想 成 一 个 excel 
表 ， 比 如 : 


char a[3][4]; 


















































假想 中 的 二 维 数组 布局 
a[1][2] 





4. 5. 1.2， 内 存 与 尺子 的 对 比 





实际 上 内 存 不 是 表 状 的 ， 而 是 线性 的 。 见 过 尺子 吧 ? 尺子 和 我 们 的 内 存 非常 相似 。 一 
般 尺 子 上 最 小 刻度 为 毫米 ， 而 内 存 的 最 小 单位 为 1 个 byte。 平 时 我 们 说 32 毫米 ， 是 指 以 零 
始 偏 移 32 毫米 ， 平 时 我 们 说 内 存 地 址 为 0x0000FF00 也 是 指 从 内 存 零 地 址 开始 偏 移 
0x0000FF00 个 byte。 既 然 内 存 是 线性 的 ， 那 二 维 数组 在 内 存 里 面 肯 定 也 是 线性 存储 的 。 实 
际 上 其 内 存 布局 如 下 图 : 




















































































































现实 中 的 二 维 数组 布局 


























以 数组 下 标的 方式 来 访问 其 中 的 某 个 元 素 : af[il[j]。 编 译 器 总 是 将 二 维 数组 看 成 是 一 个 
一 维 数 组 , 而 一 维 数 组 的 每 一 个 元 素 又 都 是 一 个 数组 。a[3] 这 个 一 维 数组 的 三 个 元 素 分 别 为 : 
a[0],a[1],a[2]。 每 个 元 素 的 大 小 为 sizeof (a[0]), 即 sizof(char)*4。 由 此 可 以 计算 出 a[0],a[1],a[2] 
三 个 元 素 的 首 地 址 分 别 为 & a[0]，& a[0]+ 1*sizof(char)*4, & a[0]+ 2*sizof(char)*4。 亦 即 a[i] 
的 首 地 址 为 & a[0]+ i*sizof(char)*4。 这 时 候 再 考虑 ai 里 面 的 内 容 。 就 本 例 面 埋 , ai 内 有 4 
个 char 类 型 的 元 素 ， 其 每 个 元 素 的 首 地 址 分 别 为 &a[i]，&a[i]+1*sizof(char)， 
&a[i]+2*sizof(char)，&a[i]+3*sizof(char)， 即 afi[j] 的 首 地 址 为 &afil+j*sizof(char)。 再 把 &afj] 






































































































































的 信用 a 表示 ， 得 到 af[il[j] 元 素 的 首 地 址 为 : a+ i*sizof(char)*4+j*sizof(char)。 同 样 ， 可 以 换 
算 成 以 指针 的 形式 表示 : *(*(ati)+j)。 
经 过 上 面 的 讲解 ， 相 信 你 已 经 掌握 了 二 维 数组 在 内 存 里 面 的 布局 了 。 下 面 就 看 一 个 题 : 
#include <stdio.h> 


int main(int argc,char * argv[]) 



























































{ 
int a [3][2]={(0,1),(2,3),(4,5)}; 
int *p; 
p=a [0]; 
printf("%d",p[0]); 
) 




















问 打印 出 来 的 结果 是 多 少 ? 

很 多 人 都 觉得 这 太 简 单 了 ， 很 快 就 能 把 答案 告诉 我 : 0。 不 过 很 可 惜 ， 错 了 。 答 案 应 该 
是 1。 如 果 你 也 认为 是 0， 那 你 实在 应 该 好 好 看 看 这 个 题 。 花 括号 里 面 蔡 套 的 是 小 括号 ， 而 
不 是 花 括 号 ! 这 里 是 花 括号 里 面 藤 套 了 过 号 表达 式 ! 其 实 这 个 赋值 就 相当 于 inta [3][2]={ 1,3, 
5); 

所 以 ， 在 初始 化 二 维 数组 的 时 候 一 定 要 注意 ， 别 不 小 心 把 应 该 用 的 花 括 号 写成 小 括号 
Ta 























































































































4.5.1.3, &p[4] [2] - &a[4] [2] 的 值 为 多 少 ? 























上 面 的 问题 似乎 还 比较 好 理解 ， 下 面 再 看 一 个 例子 : 

int a[5][5]; 

int DMA; 

p=a; 

问 &p[4][2] - &a[4][2] 的 值 为 多 少 ? 
这 个 问题 似乎 非常 简单 , 但 是 几乎 没有 人 答对 了 。 我 们 可 以 先 写 代码 测试 一 下 其 值 ， 然 后 分 
析 一 下 到 底 是 为 什么 。 在 Visual C++6.0 里 ， 测 试 代码 如 下 : 









































` 








int main() 
{ 
int a[5][5]; 
int (*p)l4]; 
p=a; 
printf("a_ptr=%#p,p_ptr=%#pn",&a[4][2],&p[41][2]); 
printf("%p, %d\n",&p[4][2] - &a[4][2],&p[41[2] - &a[4][2]); 


return 0; 


) 
经 过 测试 ， 可 知 &p[4][2] - &a[4][2] 的 值 为 -4。 这 到 底 是 为 什么 呢 ?” 下 面 我 们 就 来 分 析 一 下 : 























前 面 








如 此 ， 则 


示 的 是 













































































我 们 讲 过 ， 当 数组 名 a 作为 右 值 时 ,代表 的 是 数组 首 元 素 的 首 地 址 。 这 里 的 a 为 二 





























维 数 组 ,我 们 把 数组 a 看 作 是 包含 5 个 int 类 型 元 素 的 一 维 数组 ,里 面 再 存储 了 一 个 一 维 数组 。 























a 在 这 里 代表 的 是 a[0] 的 首 地 址 。a+l 表示 的 是 一 维 数组 a 的 第 二 介 元素 。a[4] 表 
维 数组 a 的 第 5 个 元 素 ， 而 这 个 元 素 里 又 存 了 一 个 一 维 数组 。 所 以 &af[4][2] 表 示 的 
































是 &a[0][0]+4*5*sizeof(int) + 2*sizeofGnt) 。 



































民 据 定义 ，p 是 指向 一 个 包含 4 个 元 素 的 数组 的 指针 。 也 就 是 说 p+1 表示 的 是 指针 p 向 
后 移动 了 一 个 “包含 4 个 int 类 型 元 素 的 数组 ”。 这 里 1 的 单位 是 p 所 指向 的 空间 ， 即 








4*sizeof(int)。 所 以 , p[4] 相 对 于 p[0] 来 说 是 向 后 移动 了 4 个 “包含 4 个 int 类 型 元 素 的 数组 ” 

















即 &p[4] 表 示 的 是 &p[0]+4*4*sizeof(int)。 由 于 p 被 初始 化 为 &a[0]， 那 么 &p[4[2] 表 示 的 是 
&a[0][0]+4*4*sizeofüint)+2* sizeof(int)。 



































和 由 上 面 的 讲述 ，&p[4][2] 和 &a[4][2] 的 值 相 差 4 个 int 类 型 的 元 素 。 现在， 上面 测试 














出 来 的 结果 也 可 以 理解 了 吧 ? 其 实 我 们 最 简单 的 办 法 就 是 画 内 存 布 局 图 : 





























int a[5] [5]; int (*p)[4]; a PHII21 al4]I21 
p=a; 
a[0] | a[l] | al2] | a[3] a[4 








plOI pii] p[2] p[3] pl4] p[S] 


























这 里 最 重要 的 一 点 就 是 明白 数组 指针 p 所 指向 的 内 存 到 底 是 什么 。 解 决 这 类 问题 的 最 





4.5.2 


4. 5. 2. 








， 二 级 指针 


1， 二 级 指针 的 内 存 布局 

















二 级 指针 是 经 常用 到 的 ， 尤 其 与 二 维 数组 在 一 起 的 时 候 更 是 令 人 迷糊 。 例 如 : 


char **p; 
定义 了 一 个 二 级 指针 变量 p。p 是 一 个 指针 变量 ， 毫 无 疑问 在 32 位 系统 下 占 4 个 byte。 
它 与 一 级 指针 不 同 的 是 ， 一 级 指针 保存 的 是 数据 的 地 址 ， 二 级 指针 保存 的 是 一 级 指针 的 地 


址 。 下 



































图 帮助 理解 : 








char xxp char *p2 char c 


こら こと ここ | ビー 
p= &p2 | p2=&c H---- Š, | 
F s ms 


4byte 4byte lbyte 


我 们 试 着 给 变量 p 初始 化 : 
A)，p =NULL; 








B), char *p2:p = &p2; 


任何 指针 变量 都 可 以 被 初始 化 为 NULL (注意 是 NULL， 不 是 NUL， 更 不 是 null)， 二 
级 指针 也 不 例外 。 也 就 是 说 把 指针 指向 数组 的 零 地 址 。 联 想到 前 面 我 们 把 尺子 比 作 内 存 ， 
如 果 把 内 存 初始 化 为 NULL， 就 相当 于 把 指针 指向 尺子 上 0 上 毫米 处 , 这 时 候 指针 没有 任何 内 
存 可 用 。 
当 我 们 真正 需要 使 用 p 的 时 候 ， 就 必须 把 一 个 一 级 指针 的 地 址 保存 到 p 中 , 所 以 B) 的 
赋值 方式 也 是 正确 的 。 
给 p 赋值 没有 问题 ， 但 怎么 使 用 p 呢 ? 这 就 需要 我 们 前 面 多 次 提 到 的 钥匙 (“*”)。 
第 一 步 : 根据 p 这 个 变量 ， 取 出 它 里 面 存 的 地 址 。 
第 二 步 : 找到 这 个 地 址 所 在 的 内 存 。 
第 三 步 : 用 钥匙 打开 这 块 内 存 ， 取 出 它 里 面 的 地 址 ，*p 的 值 。 
第 四 步 : 找到 第 二 次 取出 的 这 个 地 址 。 
第 五 歩 : 用 钥匙 打开 这 块 内 存 , 取出 它 里 面 的 内 容 , 这 就 是 我 们 真正 的 数据 ，**p 的 值 。 
我 们 在 这 里 用 了 两 次 钥匙 (“*”) 才 最 终 取 出 了 真正 的 数据 。 也 就 是 说 要 取出 二 级 指针 
所 真正 指向 的 数据 ， 需 要 使 用 两 次 两 次 钥匙 (“*”)。 


至 于 超过 二 维 的 数组 和 超过 二 维 的 指针 一 般 使 用 比较 少 ， 而 且 按 照 上 面 的 分 析 方 法 同 
样 也 可 以 很 轻松 的 分 析 明 白 ， 这 里 就 不 再 详细 讨论 。 读 者 有 兴趣 的 话 ， 可 以 研究 研究 。 
































































































































































































































4. 6， 数 组 参数 与 指针 参数 




















我 们 都 知道 参数 分 为 形 参 和 实 参 。 形 参 是 指 声明 或 定义 函数 时 的 参数 ， 而 实 参 是 在 调 
用 函数 时 主 调 函 数 传递 过 来 的 实际 值 。 

















4. 6.1， 一 维 数组 参数 


4. 6.1.1， 能 否 向 函数 传递 一 个 数组 ? 


看 例子 : 


void fun(char a[10]) 
{ 


char c = a[3]; 





int main() 


{ 
charb[10]= “abcdefg ; 
fun(b[10]); 
return 0: 

) 





先 看 上 面 的 调用 ，fun(b[10]); 将 b[10] 这 个 数组 传递 到 fun 函数 。 但 这 样 正确 吗 ? b[10] 
是 代表 一 个 数组 吗 ? 

显然 不 是 ， 我 们 知道 b[0] 代 表 是 数组 的 一 个 元 素 ， 那 b[10] 又 何尝 不 是 呢 ? 只 不 过 这 里 
数组 越界 了 ， 这 个 b[10] 并 不 存在 。 但 在 编译 阶段 ,编译 器 并 不 会 真正 计算 b[10] 的 地 址 并 取 
值 ， 所 以 在 编译 的 时 候 编译 器 并 不 认为 这 样 有 和 错误。 虽然 没有 错误 ， 但 是 编译 器 仍然 给 出 
了 两 个 警告 : 


warning C4047: function : char *' differs in levels of indirection from char ' 




















TH 

































































warning C4024: 'fun' : different types for formal and actual parameter 1 

这 是 什么 意思 呢 ? 这 两 个 警告 告诉 我 们 ， 函 数 参数 需要 的 是 一 个 char* 类 型 的 参数 ， 而 
实际 参数 为 char 类 型 ， 不 匹配 。 虽 然 编译 器 没有 给 出 错误 ， 但 是 这 样 运行 肯定 会 有 问题 。 
如 图 : 





























Wicrosoft Visual C++ 









































这 是 一 个 内 存 异 常 ， 我 们 分 析 分 析 其 原因 。 其 实 这 里 至 少 有 两 个 严重 的 错误 。 

第 一 : b[10] 并 不 存在 ， 在 编译 的 时 候 由 于 没有 去 实际 地 址 取 值 ， 所 以 没有 出 错 ， 但 是 
行 时 ， 将 计算 b[101 的 实际 地 址 ， 并 且 取 值 。 这 时 候 发 生 越 界 错误 。 
第 二 : 编译 器 的 警告 已 经 告诉 我 们 编译 器 需要 的 是 一 个 char* 类 型 的 参数 ， 而 传递 过 去 
的 是 一 个 char 类 型 的 参数 ， 这 时 候 fun 函数 会 将 传 入 的 char 类 型 的 数据 当地 址 处 理 ， 同 样 
会 发 生 错 误 。( 这 点 前 面 已 经 详细 讲解 ) 

第 一 个 错误 很 好 理解 , 那么 第 二 个 错误 怎么 理解 呢 ? fun 函数 明明 传递 的 是 一 个 数组 啊 ， 
编译 器 怎么 会 说 是 char* 类 型 呢 ? 别 急 ， 我 们 先 把 函数 的 调用 方式 改变 一 下 ; 

fun(b); 

b 是 一 个 数组 ， 现 在 将 数组 b 作为 实际 参数 传递 。 这 下 该 没有 问题 了 吧 ? 调试 、 运 行 ， 
一 切 正常 ， 没 有 问题 ， 收 工 ! 很 轻易 是 吧 ? 但 是 你 确认 你 真正 明白 了 这 是 怎么 回 事 ? 数组 b 


















































在 运 






















































































真 的 传递 到 了 函数 内 部 ? 
4. 6. 1.2， 无 法 向 函数 传递 一 个 数组 


我 们 完全 可 以 验证 一 下 : 
void fun(char a[10]) 
{ 


int i= sizeof (a); 
char c = a[3]; 
) 


如 果 数 组 b 真正 传递 到 函数 内 部 ， 那 i 的 值 应 该 为 10。 但 是 我 们 测试 后 发 现 i 的 值 竟然 
为 4! 为 什么 会 这 样 呢 ? 难道 数组 b 真 的 没有 传递 到 函数 内 部 ? 是 的 ， 确 实 没 有 传递 过 去 ， 
这 是 因为 这 样 一 条 规则 : 


C 语言 中 ， 当 一 维 数组 作为 函数 参数 的 时 候 ， 编 译 器 总 是 把 它 解析 成 一 个 指向 其 首 元 
素 首 地 址 的 指针 。 


这 么 做 是 有 原因 的 。 在 C 语言 中 ， 所 有 非 数 组 形式 的 数据 实 参 均 以 传 值 形式 〈 对 实 参 
做 一 份 找 贝 并 传递 给 被 调用 的 函数 ， 函 数 不 能 修改 作为 实 参 的 实际 变量 的 值 ， 而 只 能 修改 
传递 给 它 的 那 份 拷 贝 ) 调用。 然而 ， 如 果 要 拷贝 整个 数组 ， 无 论 在 空间 上 还 是 在 时 间 上 ， 
其 开销 都 是 非常 大 的 。 更 重要 的 是 ， 在 绝 大 部 分 情况 下 ， 你 其 实 并 不 需要 整个 数组 的 拷贝 ， 
你 只 想 告 诉 函 数 在 那 一 刻 对 哪个 特定 的 数组 感 兴趣 。 这 样 的 话 ， 为 了 节省 时 间 和 空间 ， 提 
高 程序 运行 的 效率 ， 于 是 就 有 了 上 述 的 规则 。 同 样 的 ， 函 数 的 返回 值 也 不 能 是 一 个 数组 ， 
而 只 能 是 指针 。 这 里 要 明确 的 一 个 概念 就 是 : 函数 本 身 是 没有 类 型 的 ， 只 有 函数 的 返回 值 
才 有 类 型 。 很 多 书 都 把 这 点 弄 错 了 ， 其 至 出 现 “XXX 类 型 的 函数 ”这 种 说 法 。 简 直 是 荒唐 
至 极 ! 


经 过 上 面 的 解释 ， 相 信 你 已 经 理解 上 述 的 规定 以 及 它 的 来 由 。 上 面 编译 器 给 出 的 提示 ， 
说 函数 的 参数 是 一 个 char* 类 型 的 指针 ， 这 点 相信 也 可 以 理解 。 


既然 如 此 ， 我 们 完全 可 以 把 fun 函数 改写 成 下 面 的 样子 : 





























































































































































































































































































































void fun(char *p) 
( 


charc = p[3];// 或 者 是 char c = *(p+3); 
} 
同样 ， 你 还 可 以 试 试 这 样子 : 

void fun(char a[10]) 

{ 











char c = a[3]; 


int main() 


{ 
charb[100]= “abcdefg”; 
fun(b): 
return 0: 

} 





运行 完全 没有 问题 。 实 际 传递 的 数组 大 小 与 函数 形 参 指定 的 数组 大 小 没有 关系 。 既 然 
如 此 ， 那 我 们 也 可 以 改写 成 下 面 的 样子 : 

void fun(char af[]) 

{ 














char c = a[3]; 
) 
改写 成 这 样 或 许 比较 好 ， 至 少 不 会 让 人 误会 成 只 能 传递 一 个 10 个 元 素 的 数组 。 


4. 6.2， 一 级 指针 参数 


4. 6. 2.1， 能 否 把 指针 变量 本 身 传 递 给 一 个 函数 


我 们 把 上 一 节 讨 论 的 列子 再 改写 一 下 : 
void fun(char *p) 








{ 
char c = p[3];// 或 者 是 char c = *(p+3); 
} 
int main() 
{ 
char *p2 = “abcdefg”; 
fun (p2) ; 
return 0; 


} 
这 个 函数 调用 ， 真 的 把 p2 本 身 传递 到 了 fun 函数 内 部 吗 ? 


我 们 知道 p2 是 main 函数 内 的 一 个 局 部 变量 ， 它 只 在 main 函数 内 部 有 效 。( 这 里 需要 
澄清 一 个 问题 : main 函数 内 的 变量 不 是 全 局 变量 ， 而 是 局 部 变量 ， 只 不 过 它 的 生命 周期 和 



































































































































全 局 变量 一 样 长 而 已 。 全 局 变量 一 定 是 定义 在 函数 外 部 的 。 初 学 者 往往 弄 错 这 点 。) 既然 它 
是 局 部 变量 ，fun 函数 肯定 无 法 使 用 p2 的 真 身 。 那 函数 调用 怎么 办 ? 好 办 : 对 实 参 做 一 份 
拷贝 并 传递 给 被 调用 的 函数 。 即 对 p2 做 一 份 拷 贝 ， 假设 其 找 贝 名 为 _ p2。 那 传递 到 函数 内 
部 的 就 是 _p2 而 并 非 p2 本 身 。 







































































4. 6. 2. 2， 无 法 把 指针 变量 本 身 传递 给 一 个 函数 


这 很 像 孙悟空 拔 下 一 根 猴 毛 变 成 自己 的 样子 去 忽悠 小 妖怪 。 所 以 fun 函数 实际 运行 时 ， 
用 到 的 都 是 _p2 这 个 变量 而 非 p2 本 身 。 如 此 ， 我 们 看 下 面 的 例子 : 


void GetMemory (char * p, int num) 


























{ 

= (char *)malloc(num*sizeof(char)); 
) 
int main() 


{ 
char *str= NULL; 


GetMemory (str, 10) ; 


strcpy(str,” hello”); 
free (str); //free 并 没有 起 作用 ， 内 存 泄漏 























z 








return 0; 


) 

在 运行 strepy(str,”hello”) 语 句 的 时 候 发 生 错 误 。 这 时 候 观 察 str 的 值 , 发 现 仍然 为 NULL。 
也 就 是 说 str 本 身 并 没有 改变 ， 我 们 malloc 的 内 存 的 地 址 并 没有 赋 给 str， 而 是 赋 给 了 _str。 
而 这 个 _str 是 编译 器 自动 分 配 和 回收 的 , 我 们 根本 就 无 法 使 用 。 所 以 想 这 样 获取 一 块 内 存 是 
不 行 的 。 那 怎么 办 ?两 个 办 法 : 


年: jJ returno 































































































char * GetMemory (char * p, int num) 


{ 
p = (char *$)malloc(num*sizeof(char)): 


return p; 


int main() 


( 


char *str= NULL; 


str = GetMemory (str, 10) ; 






























































strcpy(str, hello”); 
free Cstr); 
return 0; 
} 
这 个 方法 简单 ， 容 易 理 解 。 
第 二 : 用 二 级 指针 。 
void GetMemory (char ** p, int num) 
{ 
*p = (char *)malloc(num*sizeof(char)): 
return p; 
) 
int main() 
{ 
char *str= NULL; 
GetMemory (&str, 10) ; 
strcpy(str, hello”); 
free Cstr); 
return 0; 
} 
注意 ， 这 里 的 参数 是 &str 而 非 str。 这 样 的 话 传递 过 去 的 是 str 的 地 址 ， 是 一 个 值 。 在 函 









































数 内 部 ， 用 钥匙 〈“*2”) 来 开锁 : *(&str)， 其 值 就 是 str. 所以 malloc 分 配 的 内 存 地 址 是 真正 
赋值 给 了 str 本 身 。 


另外 关于 malloc 和 free 的 具体 用 法 ， 内 存 管理 那 章 有 详细 讨论 。 





















































4. 6. 3， 二 维 数组 参数 与 二 维 指针 参数 

















前 面 详细 分 析 了 二 维 数 组 与 二 维 指针 ， 那 它们 作为 参数 时 与 不 作为 参数 时 又 有 什么 区 
mj ? 看 例子 : 


void fun (char a[3][4]) : 




















我 们 按照 上 面 的 分 析 ,， 完全 可 以 把 af[3][4] 理 解 为 一 个 一 维 数组 af3]， 其 每 个 元 素 都 是 一 
條 含有 4 个 char 类 型 数据 的 数组 。 上 面 的 规则 ,“C 语言 中 ， 当 一 维 数组 作为 函数 参数 的 时 
候 ， 编 译 器 总 是 把 它 解析 成 一 个 指向 其 首 元 素 首 地 址 的 指针 。” 在 这 里 同样 适用 ， 也 就 是 说 
我 们 可 以 把 这 个 函数 声明 改写 为 : 

void fun (char (*p)[4]) : 

这 里 的 括号 绝对 不 能 省 略 ， 这 样 才能 保证 编译 器 把 p 解析 为 一 个 指向 包含 4 个 char 类 
型 数据 元 素 的 数组 ， 即 一 维 数组 a[3] 的 元 素 。 

同样 ， 作 为 参数 时 ， 一 维 数 组 “[]” 号 内 的 数字 可以 省略 : 

void fun (char a[ ][4]) : 
不 过 第 二 维 的 维 数 却 不 可 省 略 ， 想 想 为 什么 不 可 以 省 略 ? 

注意 : 如 果 把 上 面 提 到 的 声明 void fun(char (*p)[4) 中 的 括号 去 掉 之 后 , 声明 “void fun 
(char *p[4])” 可 以 改写 成 : 























































































































void fun (char **p) ; 
这 是 因为 参数 *p[4]， 对 于 p 来 说 ， 它 是 一 个 包含 4 个 指针 的 一 维 数 组 ， 同 样 把 这 个 一 维 数 
组 也 改写 为 指针 的 形式 ， 那 就 得 到 上 面 的 写法 。 
上 面 讨 论 了 这 么 多 ， 那 我 们 把 二 维 数组 参数 和 二 维 指 针 参 数 的 等 效 关系 整 理 一 下 : 















































数组 参数 等 效 的 指针 参数 
数组 的 数组 : char a[3][4] 数组 的 指针 : char (*p)[10] 
指针 数组 : char *a[5] 指针 的 指针 : char **p 














这 里 需要 注意 的 是 : C 语言 中 ， 当 一 维 数组 作为 函数 参数 的 时 候 ， 编 译 器 总 是 把 它 解析 
成 一 个 指向 其 首 元 素 首 地 址 的 指针 。 这 条 规则 并 不 是 递归 的 ， 也 就 是 说 只 有 一 维 数组 才 是 
如 此 ， 当 数组 超过 一 维 时 ， 将 第 一 维 改写 为 指向 数组 首 元 素 首 地 址 的 指针 之 后 ， 后 面 的 维 
再 也 不 可 改写 。 比 如 : a[3][4][5] 作 为 参数 时 可 以 被 改写 为 Cp) [4][5]。 


至 于 超过 二 维 的 数组 和 超过 二 级 的 指针 ,由 于 本 身 很 少 使 用 , 而 且 按照 上 面 的 分 析 方法 
也 能 很 好 的 理解 ， 这 里 就 不 再 详细 讨论 。 有 兴趣 的 可 以 好 好 研究 研究 。 


























ン 





















































函数 指针 


4.7.1， 函 数 指针 的 定义 





顾名思义 ， 函 数 指针 就 是 函数 的 指针 。 它 是 一 个 指针 ， 指 向 一 个 函数 。 看 例子 : 


A), char* (*funl)(char * pl,char * p2); 

















B), char* *fun2(char * pl,char * p2); 


C), char * fun3(char* pl,char * p2); 








看 看 上 面 三 个 表达 式 分 别 是 什么 意思 ? 


C): 这 很 容易 , fun3 是 函数 名 , pl, p2 是 参数 , 其 类 型 为 char* 型 , 函数 的 返回 值 为 char * 











































































































B): 也 很 简单 ， 与 C) 表达 式 相 比 ， 唯 一 不 同 的 就 是 函数 的 返回 值 类 型 为 char**， 是 个 
二 级 指针 。 

A): fun1 是 函数 名 吗 ? 回忆 一 下 前 面 讲解 数组 指针 时 的 情形 。 我 们 说 数组 指针 这 么 定 
义 或 许 更 清晰 : 

int (*)[10] p; 

再 看 看 A) 表达 式 与 这 里 何其 相似 ! 明白 了 吧 。 这 里 fun1 不 是 什么 函数 名 ， 而 是 一 人 
指针 变量 ， 它 指向 一 个 函数 。 这 个 函数 有 两 个 指针 类 型 的 参数 ， 函 数 的 返回 值 也 是 一 个 指 
针 。 同 样 ， 我 们 把 这 个 表达 式 改 写 一 下 : char* (*)(char* pl,char* p2) funl; 这 样子 是 不 
是 好 看 一 些 呢 ? 只 可 惜 编译 器 不 这 么 想 。^_^。 







































































































































































4.7.2, 函数 指針 的 使用 


4.7.2.1, 函数 指針 使用 的 例 子 





上 面 我 们 定义 了 一 个 函数 指针 ， 但 如 何 来 使 用 它 昵 ? 先 看 如 下 例子 : 
#include <stdio.h> 
#include <string.h> 








char * fun(char * pl,char * p2) 


{ 
inti = 0; 
i = stremp(p1,p2); 
if (0 == i) 
{ 
return pl; 
) 
else 
{ 
return D2: 
) 
) 
int main() 
{ 


char * (*pf) (char * pl,char * p2); 
pf = &fun; 
(*pf) ("aa","bb"); 





























} 
我 们 


也 如 此 。 
Visual C++6.0 里 ， 


return 0; 


使 用 指针 的 
































时 候 ， 需 要 通过 钥匙 (“*”) 来 取 














4.2.7.2, *(int*)&p 一 -一 这 是 什么 ? 








也 许 上 面 的 例子 过 于 简单 ， 我 们 看 看 下 面 的 例子 : 

















void 


{ 


} 


Function() 


printf("Call 


int main() 


{ 

















Function\\n"); 


void (*p)0; 
*(int*)&p=(int)Function; 


(*p) O; 


return 0; 











别 急 ， 
void 
这 行 


在 干什么 ? 
先 看 这 行 
(*p)O; 

尺码 定义 了 











&p 是 求 指 针 变 量 
(int*)&p 表示 将 ] 
(int)Function 表示 


分 析 


针 变 量 p。 


那么 
讲解 











*(int*)&p=(int)Function; 表 示 什 么 意思 ? 


代码 : 


一 个 指针 变量 p， p 指 同一 个 函数 , 这 























指向 的 内 存 里 面 的 値 , 函数 指針 合 
通过 NSK SN 然后 调用 它 。 这 里 需要 注意 到 是 ， 在 
给 函数 指针 赋值 时 ， 可 以 用 &fun E 或 直接 用 函数 名 fun。 这 是 因为 函数 名 被 
编译 之 后 其 实 就 是 一 个 地 址 ， 所 以 这 里 两 种 用 法 没有 本 质 的 差别 。 这 个 例子 很 简单 ， 就 不 再 
详细 讨论 了 。 









































这 个 函数 的 参数 和 返回 值 都 是 void。 


























到 这 里 ， 相 























(*p) 0; 束 是 


示 对 函数 的 调用 。 














量 p 本 身 的 地 址 ， 这 是 一 个 32 位 的 二 进 制 
好 址 强制 转换 成 指向 int 类 型 数据 的 指针 。 
R 将 函数 的 入 口 地 址 强制 转换 成 int 类 型 的 
音 你 已 经 明白 *(inbtt)&p=(Gin0Function; 表 示 将 函数 的 入 口 地 址 赋值 给 指 

















常数 (32 位 系统 )。 























到 这 里 ， 相 信 你 已 经 明白 了 。 其 实 函数 指针 与 普通 指针 没什么 差别 ， 只 是 指向 的 内 
容 不 同 而 已 。 

















使 
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容易 后 期 的 维护 ， 
合 


以 及 


4.7.3, 








使 接口 与 实 








系统 结构 更 加 
































现 分 开 。 


(*(void(*) 0)0) 0 ——— 这 是 




















是 不 是 感觉 上 

















摆 的 例子 太 简 单 ， 不 够 刺激 ? 好 ， 


(*(void(*) 0)0)0: 














什么 ? 


那 就 来 点 





j 函 数 指针 的 好 处 在 于 ， 可 以 将 实现 同一 功能 的 多 个 模块 统一 起 来 标识 ,这样 一 来 更 
青 晰 。 或 者 归纳 为 : 便于 分 层 设计 、 利 于 系统 抽象 、 降 低 耦 









































刺激 的 ， 看 下 面 这 个 例子 : 




















这 是 《C Traps and Pitfalls》 这 本 经 典 的 书 中 的 一 个 例子 。 没 有 发 狂 吧 ? 下 面 我 们 就 来 分 
析 分 析 : 

第 一 步 : void(*) O, 可 以 明白 这 是 一 个 函数 指针 类 型 。 这 个 函数 没有 参数 , 没有 返回 值 。 

第 二 步 : (void(*) 0)0， 这 是 将 0 强制 转换 为 函数 指针 类 型 ，0 是 一 个 地 址 ， 也 就 是 说 一 
个 函数 存在 首 地 址 为 0 的 一 段 区 域内 。 

第 三歩 : (*(void(*) 0)0)， 这 是 取 0 地址 开始 的 一 段 内 存 里 面 的 内 容 ， 其 内 容 就 是 保存 
在 首 地 址 为 0 的 一 段 区 域内 的 函数 。 

第 四 步 : (*(void(*) 0)0)0， 这 是 函数 调用 。 


好 像 还 是 很 简单 是 吧 ， 上 面 的 例子 再 改写 改写 : 








































































































(char**(*) (char **,char **))0) ( char **,char **); 
如 果 没 有 上 面 的 分 析 ， 肯 怕 不 容易 把 这 个 表达 式 看 明白 吧 。 不 过 现在 应 该 是 很 简单 的 
一 件 事 了 。 读 者 以 为 呢 ? 


と 
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llin] 


4.7.4， 函 数 指针 数组 


上 所. > 


现在 我 们 清楚 表达 式 “char* (*pf)(char* p)” 定 义 的 是 一 个 函数 指针 pf。 既 然 pf 是 
个 指针 ， 那 就 可 以 储存 在 一 个 数组 里 。 把 上 式 修改 一 下 : 
































char * (*pf[3])(char * p); 
这 是 定义 一 个 函数 指针 数组 。 它 是 一 个 数组 ， 数 组 名 为 pf， 数组 内 存储 了 3 个 指向 函数 的 
指针 。 这 些 指针 指向 一 些 返 回 值 类 型 为 指向 字符 的 指针 、 参 数 为 一 个 指向 字符 的 指针 的 函 


数 。 这 念 起 来 似乎 有 点 掏 口 。 不 过 不 要 紧 ， 关 键 是 你 明白 这 是 一 个 指针 数组 ， 是 数组 。 










































































函数 指针 数组 怎么 使 用 呢 ? 这 里 也 给 出 一 个 非常 简单 的 例子 , 只 要 真正 掌握 了 使 用 方法 ， 
再 复杂 的 问题 都 可 以 应 对 。 如 下 : 


#include <stdio.h> 


























#include <string.h> 
char * funl(char * p) 
{ 

printf("%s\n",p); 


return p; 


char * fun2(char * p) 
{ 
printf("%s\n",p); 


return p; 


char * fun3(char* p) 
{ 
printf("%s\n",p); 


return D: 


int main() 
{ 
char * (*pf[3])(char * p); 
pf[0] = fun1; // 可 以 直接 用 函数 名 
pf[1] = &fun2; // 可 以 用 函数 名 加 上 取 地 址 符 
pf[2] = &fun3: 




















pf[0]("fun1"); 
pf[0]("fun2"); 
pf[0]("fun3"); 


return 0: 


4.7.5， 函 数 指针 数组 的 指针 


看 着 这 个 标题 没 发 狂 吧 ? 函数 指针 就 够 一 般 初 学 者 折腾 了 ， 函 数 指 针 数 组 就 更 加 麻烦 ， 
现在 的 函数 指针 数组 指针 就 更 难 理解 了 。 
RK, 没 这 么 复杂 。 前 面 详细 讨论 过 数组 指针 的 问题 , 这 里 的 函数 指针 数组 指针 不 就 是 
一 个 指针 嘛 。 只 不 过 这 个 指针 指向 一 个 数组 ， 这 个 数组 里 面 存 的 都 是 指向 函数 的 指针 。 仪 
此 而 已 。 
下 面 就 定义 一 个 简单 的 函数 指针 数组 指针 : 
char * (*(*pf)[3])(char * p); 
注意 , 这 里 的 pf 和 上 一 节 的 pf 就 完全 是 两 码 事 了 。 上 一 节 的 pf 并 非 指针 , 而 是 一 个 数组 名 ; 
这 里 的 pf 确实 是 实 实在 在 的 指针 。 这 个 指针 指向 一 个 包含 了 3 个 元 素 的 数组 ;这 个 数字 里 
面 存 的 是 指向 函数 的 指针 ， 这 些 指针 指向 一 些 返 回 值 类 型 为 指向 字符 的 指针 、 参 数 为 一 个 
指向 字符 的 指针 的 函数 。 这 比 上 一 节 的 函数 指针 数组 更 抛 口 。 其 实 你 不 用 管 这 么 多 ， 明 白 
这 是 一 个 指针 就 ok 了 。 其 用 法 与 前 面 讲 的 数组 指针 没有 差别 。 下 面 列 一 个 简单 的 例子 : 
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#include <stdio.h> 


#include く string.h> 


char * funl(char * p) 
{ 
printf("%s\n",p); 


return D: 


char * fun2(char * p) 
{ 
printf("%s\n",p); 


return p; 


char * fun3(char * p) 

{ 
printf("%s\n",p); 
return p; 

) 

int main() 

{ 
char * (*a[3]) (char * p); 
char * (*(*pf)[3])(char * p); 
pf = &a; 


a[0] = funl; 
a[l] = &fun2; 
al2] = &fun3; 


pf[O][OICfan1): 
pfl0][1]('fun2"); 


pfl0][2]("fun3"); 


return 0; 











欢迎 您 进入 这 片 雷 区 。 我 欣赏 
怕 ， 不 留 地 雷 在 人 间 ” 的 勇者 。 i 

















心态 取胜 。 














第 五 章 内 存 管理 


Q 





T 


曾经 很 短暂 的 使 用 过 一 段 时 间 的 C#。 头 三 天 特别 不 习惯 ， 因 





a 


能 活着 走 出入 上 
您 不 


把 这 当 作 一 个 扫雷 游戏 ， 


















































x 的 高 手 ， 但 更 欣 党 “ 粉 身 碎 骨 浑 不 
因为 没有 人 能 以 游戏 的 





为 没有 指针 ! 后 来 用 起 来 


越 来 越 顺 手 ， 还 是 因为 没有 指针 ! 几 天 的 时 间 很 轻易 的 写 了 1 万 多 行 C# 代 码 ， 感 觉 比 用 C 


























或 C++ 简单 多 了 。 因 为 你 根本 就 不 用 去 考虑 底层 的 





























内 存 管理 ， 也 不 用 考虑 内 存 泄漏 的 问题 ， 


更 加 不 怕 “ 野 指针 ”( 有 的 书 叫 “ 最 垂 指针 ”)。 所 有 这 一 切 ， 系 统 都 给 你 做 了 ， 所 以 可 以 很 


























轻松 的 拿 来 就 用 。 但 是 C 或 Ct++， 这 一 切 都 必须 你 自己 来 处 理 ， 即 使 经 验 
不 了 犯错 。 我 曾经 做 过 一 个 ] 





vo 




















富 的 老手 也 免 











MH, HA 


























这 个 bug 很 少 出 现 ， 但 是 




















5. 1， 什 么 是 野 指针 


那 到 底 什么 是 野 指 针 呢 ? 怎么 去 理解 这 个 “ 野 ” 呢 ? RATI 


词 : 





W£ f: 没 人 要 ， 没 人 管 
野 狗 : 没有 主人 的 狗 ， 没 有 链子 锁 着 的 狗 ， 
对 付 野 孩子 的 最 好 办 法 是 给 他 定 





好 收拾 他 。 对 付 野 狗 最 好 的 办 法 就 是 拿 条 狗 链 锁 着 它 ， 不 让 它 四 处 乱 跑 。 


对 付 也 指针 表 怕 比 对 付 野 孩 子 或 野 狗 更 困难 。 我 们 需要 把 对 付 野 孩子 和 野 狗 的 办 法 都 
用 上 。 既 需要 规 抵 ， 也 需要 链子 。 


前 面 我 们 把 内 存 比 作 尺子 ， 很 轻松 的 到 














的 同时 最 好 初始 化 为 NULL， 


























的 孩子 ， 行 为 动作 不 守 规 外 




















































































































喜欢 四 处 咬 人 。 


E， 调 皮 揭 蛋 的 孩子 。 








套 规 矩 ， 好 好 管教 。 一 旦 发 现 没 有 按 规矩 办 事 就 好 














E 解 了 内 存 。 尺 子 上 的 0 毫米 处 就 是 内 存 的 0 地 
址 处 ， 也 就 是 NULL 地 址 处 。 这 条 栓 “ 时 指针 ”的 链子 就 是 这 个 “NULL”。 定 义 指针 变量 
j 完 指针 之 后 也 将 指针 变量 的 值 设 置 为 NULL。 








在 使 用 时 ， 别 的 时 间 都 把 指针 “ 栓 ” 到 0 地 址 处 。 这 样 它 就 老实 了 。 


5.2， 栈 、 堆 和 静态 区 


对 于 程序 员 ， 一 般 来 说 ， 我 们 可 以 简 
民 多 书 没有 把 把 堆 和 栈 解释 ; 
堆 的 英文 是 heap， 栈 的 英文 是 stack， 也 翻译 为 
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老林 


月 E> 导致 初学 者 总 
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时 解 为 内 存 分 为 三 个 部 分 : 

















SK, R H. 













































































是 分 不 清楚 。 其 实 堆 栈 就 是 栈 ， 而 不 是 堆 。 
任 栈 。 堆 和 栈 都 有 自己 的 特性 ， 这 里 先 不 做 


F} 提 交 给 客户 很 久之 后 ， 客 户 发 现 一 个 很 严重 的 bug. 
旦 出 现 就 是 致命 的 ， 系 统 无 法 启动 ! 这 个 问题 交 给 我 来 解决 。 
由 于 要 再 现 这 个 bug 十 分 困难 ， 按 照 客户 给 定 的 操作 步骤 根本 无 法 再 现 。 经 过 大 概 2 周 时 
间 天 天 和 客户 越 洋 视频 之 后 ， 终 于 找到 了 bug 的 原因 一 一 野 指針 ! 所 以 关于 内 存 管理 ， 尤 
其 是 野 指 针 的 问题 ， 干 万 千 万 不 要 掉以轻心 ， 否 则 ， 你 会 很 惨 的 。 








E 看 别 的 两 个 关于 “ 野 ” 的 














也 就 是 说 除了 

















才 论 。 再 打 个 比方 : 一 层 教学 楼 ， 可 能 有 外 语 教室 ， 人 允许 外 语系 学 生 和 老师 i 
有 数学 教师 ， 人 允许 数学 系 学 生 和 老师 进入 ; 还 可 能 有 校长 办 公 室 ， 人 允许 校长 
内 存 也 是 这 样 ， 内 存 的 三 个 部 分 ， 不 是 所 有 的 东西 都 能 存 进 去 的 。 
静态 区 : 保存 自动 全 局 变量 和 static 变量 (包括 static 全 局 和 局 部 变量 )。 静 态 
在 总 个 程序 的 生命 周期 内 都 存在 ， 由 编译 器 在 编译 的 时 候 分 配 。 

栈 : 保存 局 部 变量 。 栈 上 的 内 容 只 在 函数 的 范围 内 存在 ， 当 函数 运行 结束 ， 这 些 内 容 
也 会 自动 被 销毁 。 其 特点 是 效率 高 ， 但 空间 大 小 有 限 。 
BË: 由 malloc 系列 函数 或 new 操作 符 分 配 的 内 存 。 其 生命 周期 由 free 或 delete 决定 。 


在 没有 释放 之 前 一 直 存 在 ， 5 间 比 较 大 ， 但 容易 出 错 。 































































































































































































































































































到 程序 结束 。 其 特点 是 使 用 灵活 ， 空 间 
































5.3， 常 见 的 内 存 错 误 及 对 入 


5. 3. 1， 指 针 没有 指向 一 块 合法 的 内 存 


定义 了 指针 变量 ， 但 是 没有 为 指针 分 配 内 存 ， 即 指针 没有 指向 一 块 合法 的 内 存 。 




















浅显 的 例子 就 不 举 了 ， 这 里 举 几 个 比较 隐蔽 的 例子 。 





























5. 3. 1. 1， 结 构 体 成 员 指针 未 初始 化 


struct student 
char *name; 
Int score; 


}stu,*pstu; 


int main() 


{ 


strcpy(stu.name, "Jimy"); 
stu.score = 99; 


return 0; 


} 

很 多 初学 者 犯 了 这 个 错误 还 不 知道 是 怎么 回 事 。 这 里 定义 了 结构 体 变量 stu， 但 是 他 没 
想到 这 个 结构 体内 部 char *name 这 成 员 在 定义 结构 体 变 量 stu 时 ， 只 是 给 name 这 个 指针 变 
量 本 身分 配 了 4 个 字 节 。name 指针 并 没有 指向 一 个 合法 的 地 址 ， 这 时 候 其 内 部 存 的 只 是 一 
些 乱 码 。 所 以 在 调用 strcpy 函数 时 ， 会 将 字符 串 "Jimy" 往 乱码 所 指 的 内 存 上 拷贝 ， 而 这 块 内 


















































存 name 指针 根本 就 无 权 访问 ， 导 致 出 错 。 解 决 的 办 法 是 为 name 指针 malloc 一 块 空 间 。 








同样 ， 也 有 人 犯 如 下 错误 : 





int main() 


{ 


pstu = (struct student*)malloc(sizeof(struct student)); 


strcpy(pstu->name, "Jimy"); 


pstu->score = 99; 


free(pstu); 


return 0; 


} 





为 指针 变量 pstu 分 配 了 内 存 , 但 是 同样 没有 给 name 指针 分 配 内 存 。 错 误 与 上 面 第 一 种 








情况 一 样 , 解决 的 办 法 也 一 样 
配 了 内 存 。 















































。 这 里 用 了 一 个 malloc 给 人 一 种 错 














5. 3. 1.2， 没 有 为 结构 体 指针 分 配 足够 的 内 存 


int main() 


( 


pstu = (struct student*)malloc(sizeot(struct student*)): 


strcpy(pstu->name," Jimy"); 


pstu->score = 99; 
free(pstu); 
return 0; 


) 























觉 ， 以 为 也 给 name 指針 分 


为 pstu 分 配 内 存 的 时 候 ， 分 配 的 内 存 大 小 不 合适 。 这 里 把 sizeof(struct student) 误 写 为 
sizeof(struct student*) 。 当然 name 指针 同样 没有 被 分 配 内 存 。 解 决 办 法 同上 。 














5. 3. 1.3， 函 数 的 入 口 校 验 


不 管 什么 时 候 ， 我 们 使 用 指针 之 前 一 定 要 确保 指针 是 有 效 的 。 











一 般 在 函数 入 口 处 使 用 





if (NULL !=p) 来 校 验 。 但 ; 








如 上 面 的 例子 ， 即 使 用 过 (NULL =p) 校 验 也 起 不 了 作用 ， 因 

















这 都 有 一 个 要 求 ， 即 p 在 定义 的 同 




















E NULL 的 乱码 。 





化 为 NULL， 其 内 部 是 一 个 


assert(NULL != p) 对 参数 进行 校 验 。 在 非 参数 的 地 方 使 用 
时 被 初始 化 为 NULL 了 。 比 











为 name 指针 并 没有 被 初始 
































assert 是 一 个 宏 ， 而 不 是 函数 ， 包 含 在 asserth 头 文件 中 。 如 果 其 后 面 括号 里 的 值 为 假 ， 
则 程序 终止 运行 ， 并 提示 出 错 ， 如 果 后 面 括号 里 的 值 为 真 ， 则 继续 运行 后 面 的 代码 。 这 个 
BRE Debug 版 本 上 起 作用 ， 而 在 Release 版 本 被 编译 器 完全 优化 掉 ， 这 样 就 不 会 影响 代码 
的 性 能 。 


有 人 也 许 会 问 ， 既 然 在 Release 版 本 被 编译 器 完全 优化 掉 ， 那 Release 版 本 是 不 是 就 完 
全 没有 这 个 参数 入 口 校 验 了 呢 ?” 这 样 的 话 那 不 就 跟 不 使 用 它 效果 一 样 吗 ? 


是 的 , 使 用 assert 宏 的 地方 在 Release 版 本 里 面 确实 没有 了 这 些 校 验 。 但 是 我 们 要 知道 ， 
assert 宏 只 是 帮助 我 们 调试 代码 用 的 ， 它 的 一 切 作 用 就 是 让 我 们 尽 可 能 的 在 调试 函数 的 时 候 
把 错误 排除 掉 ， 而 不 是 等 到 Release 之 后 。 它 本 身 并 没有 除 错 功能 。 再 有 一 点 就 是 ， 参 数 出 
现 错误 并 非 本 函数 有 问题 ， 而 是 调用 者 传 过 来 的 实 参 有 问题 。assert 宏 可 以 帮助 我 们 定位 错 
误 ， 而 不 是 排除 错误 。 































































































































































































































































































5. 3. 2， 为 指针 分 配 的 内 存 太 小 

















为 指针 分 配 了 内 存 ， 但 是 内 存 大 小 不 够 ， 导 致 出 现 越界 错误 。 
char *p1 = “abcdefg”; 





char *p2 = (char *)malloc(sizeof(char)*strlen(p1)): 

strepy(p2,p1); 

pl 是 字符 串 常量 ， 其 长 度 为 7 个 字符 ， 但 其 所 占 内 存 大 小 为 8 全 byte。 初 学者 往 往 忘 
了 字符 串 常量 的 结束 标志 “\0”。 这 样 的 话 将 导致 pl 字符 串 中 最 后 一 个 空 字符 “\0” 没 有 被 
拷贝 到 p2 中 。 解 决 的 办 法 是 加 上 这 个 字符 串 结束 标志 符 : 
























































char *p2 = (char *)malloc(sizeof(char)*strlen(p1)+1*sizeof(char)); 


这 里 需要 注意 的 是 ， 只 有 字符 串 常 量 才 有 结束 标志 符 。 比 如 下 面 这 种 写法 就 没有 结束 标志 符 



































char a[7] = { DE ef e, }; 


另外 ， 不 要 因为 char 类 型 大 小 为 1 个 byte 就 省 略 sizof (char) 这 种 写法 。 这 样 只 会 使 
你 的 代码 可 移植 性 下 降 。 


























ーー 

















5.3.3, 内 存 分 配 成功 , 但 井 未 初 始 化 




















犯 这 个 错误 往往 是 由 于 没有 初始 化 的 概念 或 者 是 以 为 内 存 分 配 好 之 后 其 值 自然 为 0。 未 
初始 化 指针 变量 也 许 看 起 来 不 那么 严重 ， 但 是 它 确 确实 实 是 个 非常 严重 的 问题 ， 而 且 往往 
出 现 这 种 错误 很 难 找到 原因 。 


曾经 有 一 个 学 生 在 写 一 个 windows 程序 时 ， 想 调用 字库 的 某 个 字体 。 而 调用 这 个 字库 
需要 填充 一 个 结构 体 。 他 很 自然 的 定义 了 一 个 结构 体 变量 ， 然 后 把 他 想 要 的 字库 代码 赋值 
给 了 相关 的 变量 。 但 是 ， 问 题 就 来 了 ， 不 管 怎么 调试 ， 他 所 需要 的 这 种 字体 效果 总 是 不 出 
来 。 我 在 检查 了 他 的 代码 之 后 ， 没 有 发 现 什么 问题 ， 于 是 单 步调 试 。 在 观察 这 个 结构 体 变 































































































































































































量 的 内 存 时 ， 发 现 有 几 个 成 员 的 值 为 乱码 。 就 是 其 中 某 一 个 乱码 车 得 祸 ! 因为 系统 会 按照 
这 个 结构 体 中 的 茶 些 特定 成 员 的 值 去 字库 中 寻找 匹配 的 字体 ， 当 这 些 值 与 字库 中 某 种 字体 
的 某 些 项 匹配 时 ， 就 调用 这 种 字体 。 但 是 很 不 幸 ， 正 是 因为 这 几 个 乱码 ， 导 致 没有 找到 相 
匹配 的 字体 ! 因为 系统 并 无 法 区 分 什么 数据 是 乱码 ， 什 么 数据 是 有 效 的 数据 。 只 要 有 数据 ， 
系统 就 理所当然 的 认为 它 是 有 效 的 。 

也 许 这 种 严重 的 问题 并 不 多 见 ， 但 是 也 绝 不 能 掉以轻心 。 所 以 在 定义 一 个 变量 时 ， 第 
件 事 就 是 初始 化 。 你 可 以 把 它 初始 化 为 一 个 有 效 的 值 ， 比 如 


int 1=10; 

































































































































































char *p = (char $malloc(sizeof(char)): 


但 是 往往 这 个 时 候 我 们 还 不 确定 这 个 变量 的 初 值 ， 这 样 的 话 可 以 初始 化 为 0 或 NULL。 
































int i=0; 
char *p = NULL; 

如 果 定 义 的 是 数组 的 话 ， 可 以 这 样 初 始 化 : 
inta[10]= {0}; 

或 者 用 memset 函数 来 初始 化 为 0: 


memset (a,O,sizeof(a) ) ; 

memset 函数 有 三 个 参数 ， 第 一 个 是 要 被 设置 的 内 存 起 始 地 址 ， 第 二 个 参数 是 要 被 设置 
的 值 ， 第 三 个 参数 是 要 被 设置 的 内 存 大 小 ， 单 位 为 byte。 这 里 并 不 想 过 多 的 讨论 memset K 
数 的 用 法 ， 如 果 想 了 解 更 多 ， 请 参考 相关 资料 。 

至 于 指针 变量 如 果 未 被 初始 化 ， 会 导致 让 语句 或 assert 宏 校 验 失 败 。 这 一 点 ， 上 面 已 有 
分 析 。 






















































































5.3.4, 内 存 越 界 





内 存 分 配 成 功 ， 且 已 经 初始 化 ， 但 是 操作 越过 了 内 在 的 边界 。 
这 种 错误 经 常 是 由 于 操作 数组 或 指针 时 出 现 “多 1” 或 “ 少 1”。 比 如 : 
int a[10] = {0}; 





for (i=0; i<=10; i++) 
{ 

ali] =i; 
) 


PELL, for 循环 的 循环 变量 一 定 要 使 用 半 开 半 闭 的 区 间 ， 而 且 如 果 不 是 特殊 情况 ,循环 变 
EREA 0 开始。 
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5. 3.5， 内 存 泄漏 





























内 存 泄漏 几乎 是 很 难 避免 的 ， 不 管 是 老手 还 是 新 手 ， 都 存在 这 个 问题 。 甚 至 包括 
windows, Linux 这 类 软件 ， 都 或 多 或 少 有 内 存 汇 漏 。 也 许 对 于 一 般 的 应 用 软件 来 说 ， 这 个 
问题 似乎 不 是 那么 突出 ， 重 启 一 下 也 不 会 造成 太 大 损失 。 但 是 如 果 你 开发 的 是 嵌入 式 系 统 
软件 呢 ? 比 如 汽车 制 动 系统 ， 心 脏 起 搏 器 等 对 安全 要 求 非常 高 的 系统 。 你 总 不 能 让 心脏 起 
搏 器 重启 吧 ， 人 家 阁 王 老爷 是 非常 好 客 的 。 

会 产生 泄漏 的 内 存 就 是 堆 上 的 内 存 〈 这 里 不 讨论 资源 或 句柄 等 泄漏 情况 )， 也 就 是 说 由 
malloc 系列 函数 或 new 操作 符 分 配 的 内 存 。 如 果 用 完 之 后 没有 及 时 free 或 delete， 这 块 内 存 
就 无 法 释放 ， 直 到 整个 程序 终止 。 






































































































































5. 3. 5. 1， 告 老 还 乡 求 良田 





怎么 去 理解 这 个 内 存 分 配 和 释放 过 程 呢 ? 先 看 下 面 这 段 对话 : 
HIE: RW, MARKETE FSD, 想 要 何 賞 喝 喘 ? 


某 功臣 : JZ, 黄金 白銀 , 臣 祝 之 如 美 土 。 臣 年 少 己 老 , MEZL. BARHATE 
以 荫 后 世 ， 别 无 他 求 。 


HIE: 爱 卿 ， 你 劳苦 功 高 ， 却 仪 要 如 此 小 赏 ， 腾 今天 就 如 你 所 愿 。 户 部 刘 侍 郎 ， 查 
看 湖广 一 带 是 否 还 有 千 亩 上 等 良田 未 曾 封赏 。 


XER: 长 沙 尚 有 五 万 余 亩 上 等 良田 末 曾 封赏 。 
HIE: 在 藤 沙 援 良 田 千 再 封 賞 愛 帰 。 愛 卿 , KEFE, 作 欲 何 用 咽 ? 


某 功臣 : 谢 万 岁 。 长 沙 一 带 ， 适 合 种 水 稻 ， 臣 想 用 来 种 水 稻 。 种 水 稻 需 要 把 田 分 为 一 
亩 一 块 ， 方 便 耕 种 















































































































































5. 3. 5. 2， 如 何 使 用 malloc 函数 





























不 要 英名 其 妙 ， 其 实 上 面 这 段 小 小 的 对 话 ， 就 是 malloc 的 使 用 过 程 。malloc 是 一 个 函 
数 ， 专 门 用 来 从 堆 上 分 配 内 存 。 使 用 malloc 函数 需要 几 个 要 求 : 


内 存 分 配给 谁 ? 这 里 是 把 良田 分 配给 某 功 臣 。 
分 配 多大 内 存 ? 这 里 是 分 配 一 千 亩 。 
否 还 有 足够 内 存 分 配 ? 这 里 是 还 有 足够 良田 分 配 。 


是 
内 存 的 将 用 来 存储 什么 格式 的 数据 ， 即 内 存 用 来 做 什么 ?这 是 
分 成 一 百 一 块 。 


分 配 好 的 内 存在 哪里 ? 这 里 是 在 长 沙 。 
如 果 这 五 点 都 确定 ， 那 内 存 就 能 分 配 。 下 面 先 看 malloc 函数 的 原型 : 




























































































mm 
mm 


用 来 种 水 稻 ， 需 要 把 田 

































































(void *)malloc(int size) 





malloc 函数 的 返回 值 是 一 个 void 类 型 的 指针 ， 参 数 为 int 类 型 数据 ， 即 申请 分 配 的 内 存 
































大 小 ,单位 是 byte。 内 存 分 配 成 功 之 后 ，malloc 函数 返回 这 块 内 存 的 首 地 址 。 你 需要 一 个 指 

















针 来 接收 这 个 地 址 。 但 是 











的 类 型 。 也 就 是 说 ， 这 块 内 存 将 要 月 


char *p = (char *)malloc(100); 








于 函数 的 返回 














在 堆 上 分 配 了 100 个 字 节 内 存 ， 返 





口 





























你 只 能 通过 指针 变量 p 来 操作 这 块 
问 。 



































内 存 。 这 块 内 存 本 映 并 没有 名 字 ， 对 它 的 访问 是 匿名 访 
































值 是 void * 类 型 的 ， 所 以 必须 强制 转换 成 你 所 接收 
日 来 存储 什么 类 型 的 数据 。 比 如 : 








这 块 内 存 的 首 地 址 ， 把 地 址 强制 转换 成 char * 类 型 后 赋 
给 char* 类 型 的 指针 变量 p。 同 时 告诉 我 们 这 块 内 存 将 用 来 存储 char 类 型 的 数据 。 也 就 是 说 


















































上 面 就 是 使 用 malloc 函数 成 功 分 配 一 块 内 存 的 过 程 。 但 是 ， 每 次 你 都 能 分 配 成 功 吗 ? 


























\ 一 定 。 上 面 的 对 话 ， 皇 帝 让 户 部 侍郎 查询 是 否 还 有 足够 的 良田 末 被 分 配 出 去 。 使 用 malloc 




















函数 同样 要 注意 这 点 : 如 果 所 申请 的 内 存 块 大 于 目前 夫 
会 失败 ， 函 数 返回 NULL。 注 意 这 里 说 的 “ 夫 


















































为 malloc 函数 申请 的 是 连续 的 一 块 内 存 。 
既然 malloc 函数 申请 内 存 有 不 成 功 的 可 能 ， 




















上 剩余 内 存 块 〈 整 块 )， 则 内 存 分 配 


== 

















上 剩余 内 存 块 ” 不 是 所 有 剩余 内 存 块 之 和 ， 因 


那 我 们 在 使 用 指向 这 块 内 存 的 指针 时 ， 必 




















须 用 if (NULL ! =p) 1840360641 





5. 3. 5.3， 用 malloc 函数 申请 0 字 节 内 存 




















E 内 存 确 实 分 配 成 功 了 。 





另外 还 有 一 个 问题 : 用 malloc 函数 申请 0 字 节 内 存 会 返回 NULL 指针 吗 ? 








可 以 测试 一 下 ， 也 可 以 去 查找 关于 malloc 函数 的 说 明文 档 。 申 请 0 字 节 内 存 ， 函 数 # 


























不 返回 NULL， 而 是 返回 一 个 正常 的 内 存 ] 
好 尺子 上 的 某 个 刻度 ， 刻 度 本 身 并 没有 长 度 ， 
























































地 址 。 但 是 你 却 无 法 使 用 这 块 大 小 为 0 的 内 存 。 这 








只 有 茶 两 个 刻度 一 起 才能 量 出 长 度 。 对 于 这 





一 点 一 定 要 小 心 ， 因 为 这 时 候 让 (NULL ! =p) 语句 校 验 将 不 起 作用 。 





5. 3. 5. 4， 内 存 释 放 


既然 有 分 配 ， 那 就 必须 有 释放 。 不 然 的 话 ， 有 限 


























的 内 存 总 会 用 光 ， 而 没有 释放 的 内 存 








却 在 空闲 。 与 malloc 对 应 的 就 是 free KALI o free 函数 只 有 一 个 参数 ， 就 是 所 要 释放 的 内 


存 块 的 首 地 址 。 比 如 上 例 ; 


free(p); 














free 函数 看 上 去 挺 狠 的 , 但 它 到 底 作 了 什么 呢 ? 其 














这 块 内 存 的 关系 。 比 如 上 面 的 例子 


























实 它 就 做 了 一 件 事 : 斩 断 指针 变量 与 











我 们 可 以 说 malloc 函数 分 配 的 内 存 块 是 属于 p 的 ， 因 
为 我 们 对 这 块 内 存 的 访问 都 需要 通过 p 来 进行 。free 函数 就 是 把 这 块 内 存 和 p 之 间 的 所 有 关 








系 斩 断 。 从 此 p 和 那 块 内 存 之 间 再 无 瓜葛 。 至 于 指针 变量 p 本 身 保 存 的 地 址 并 没有 改变 ， 
但 是 它 对 这 个 地 址 处 的 那 块 内 存 却 已 经 没有 所 有 权 了 。 那 块 被 释放 的 内 存 里 面 保存 的 值 也 
























































没有 改变 ， 只 是 再 也 没有 办 法 使 



































了 。 
这 就 是 free 函数 的 功能 。 按 照 上 面 的 分 析 ， 如 果 对 p 连续 两 次 以 上 使 用 free 函数 ， 肯 


定 会 发 生 错误 。 因 为 第 一 使 用 free 函数 时 ，p 所 
































内 存 可 释放 了 。 关 于 这 点 ， 我 上 课时 让 学 生 记 住 































































































属 的 内 存 已 经 被 释放 ， 第 二 次 使 用 时 已 经 无 














的 是 : 


定 要 一 夫 一 妻 制 , APP E H +B 











malloc 两 次 只 free 一 次 会 内 存 浴 漏 , malloc 一 次 free 两 次 肯定 会 出 错 。 也 就 是 说 ， 在 程序 
中 malloc 的 使用 次 数 一 定 要 和 free 相等 ， 否 则 必 有 和 错误。 这 种 错误 主要 发 生 在 循环 使 用 
malloc 函数 时 ， 往 往 把 malloc 和 free 次 数 弄 错 了 。 这 里 留 个 练习 : 


写 两 个 函数 ， 一 个 生成 链表 ， 一 个 释放 链表 。 两 个 函数 的 参数 都 只 使 用 一 个 表 头 指针 。 




































































5. 3. 5.5， 内 存 释 放 之 后 








mh 


既然 使 用 free 函数 之 后 指针 变量 p 本 身 保存 的 地 址 并 没有 改变 , 那 我 们 就 需要 重新 把 p 
的 值 变 为 NULL: 


p = NULL; 
这 个 NULL 就 是 我 们 前 面 所 说 的 “ 栓 野 狗 的 链子 ”。 如 果 你 不 栓 起 来 迟早 会 出 问题 的 。 比 如 : 
fE free (p) 之 后 ， 你 用 if (NULL ! =p) 这 样 的 校 验 语句 还 能 起 作用 吗 ? 
例如 : 
char *p = (char *) malloc(100); 
























































strcpy(p, “hello” ); 
free(p); /#p 所 指 的 内 存 被 释放 ， 但 是 p 所 指 的 地 址 仍然 不 变 7 

















if (NULL != p) 
{ 











だ 没有 起 到 防 错 作用 */ 


strepy(p，“world”); だ 出 错 */ 











} 
释放 完 块 内 存 之 后 , 没有 把 指针 置 NULL， 这 个 指针 就 成 为 了 “ 野 指针 ” BARY 
重 指 针 ”。 这 是 很 危险 的 ， 而 且 也 是 经 常 出 错 的 地 方 。 所 以 一 定 要 记 住 一 条 : free 完 之 后 ， 
一 定 要 给 指针 置 NULL。 

同时 留 一 个 问题 : 对 NULL 指针 连续 free 多 次 会 出 错 吗 ? 为 什么 ?如 果 让 你 来 设计 free 
函数 ， 你 会 怎么 处 理 这 个 问题 ? 



























































5. 3. 6， 内 存 已 经 被 释放 了 ， 但 是 继续 通过 指针 来 使 用 


这 里 一 般 有 三 种 情况 : 

第 一 种 ， 就 是 上 面 所 说 的 ，free (p) 之 后 ， 继 续 通 过 p 指针 来 访问 内 存 。 解 决 的 办 法 
就 是 给 p 置 NULL。 

第 二 种 : 函数 返回 栈 内 存 。 这 是 初学 者 最 容易 犯 的 错误 。 比 如 在 函数 内 部 定义 了 一 个 
数组 , 却 用 return 语句 返回 指向 该 数组 的 指针 。 解 决 的 办 法 就 是 弄 明白 栈 上 变量 的 生命 周期 。 





























第 三 种 : 内 存 使 用 太 复 杂 ， 乔 不 清 




















法 是 重新 设计 程序 ， 改 善 对 象 之 间 的 调用 关系 。 




















4 到底 哪 块 内 存 被 释放 ， 哪 块 没有 被 释放 。 解 决 的 办 














8162 

















误 发 生 的 原因 及 预防 3 


讨论 了 常见 的 六 种 错误 及 解决 对 策 , 希望 读者 仔细 下 




















读 , 尽量 使 自己 对 每 种 错 











F 段 烂熟 于 胸 。 一 定 要 多 练 ， 多 调试 代码 ， 同 时 多 总 结 经 验 。 


第 六 章 函数 





什么 是 函数 ? 为 什么 需要 函数 ? 这 两 个 看 似 很 简单 的 问题 ， 你 能 回答 清楚 吗 ? 








6. 1， 函 数 的 由 来 与 好 处 

















其 实在 汇编 语言 阶段 ， 函 数 这 个 概念 还 是 比较 模糊 的 。 汇编 语言 的 代码 往往 
开始 一 条 一 条 执行 ， 直 到 遇 到 跳 转 指令 〈 比 如 ARM 指令 B、BL、BX、BLX 之 
































就 是 从 入 口 
类 ) 然后 才 














跳 转 到 目的 指令 处 执行 。 这 个 时 候 所 有 的 代码 仅仅 是 按 其 将 要 执行 的 顺序 排列 而 已 。 后 来 人 



































们 发 现 这 样 写 代码 非常 费劲 ， 容 易 出 错 ， 也 不 方便 。 于 是 想 出 一 个 办 法 ， 把 一 些 








功能 相对 来 








说 能 成 为 一 个 整体 的 代码 放 到 一 起 打包 , 通过 一 些 数据 接口 和 外 界 通信 。 这 就 是 函数 的 由 来 。 











那 函数 能 给 我 们 带 来 什么 好 处 呢 ? 简单 来 说 可 以 概括 成 以 下 几 点 : 
1、 降 低 复 杂 性 : 使 用 函数 的 最 首要 原因 是 为 了 降低 程序 的 复杂 性 ， 可 以 使 

含 信 息 ， 从 而 使 你 不 必 再 考虑 这 些 信息 。 

2、 和 避免 重复 代码 段 : 如 果 在 两 个 不 同 函 数 中 的 代码 很 相似 ， 这 往往 意味 着 

































































误 。 这 时 ， 应 该 把 两 个 函数 中 重复 的 代码 都 取出 来 ， 把 公共 代码 放 入 一 个 新 的 通用 函数 中 ， 


























用 函数 来 隐 


分 解 工作 有 











然后 再 让 这 两 个 函数 调用 新 的 通用 函数 。 通 过 使 公共 代码 只 出 现 一 次 ， 可 以 节约 许多 空间 。 





























因为 只 要 在 一 个 地 方 改动 代码 就 可 以 了 。 这 时 代码 也 更 可 靠 了 。 
3、 限 制 改 动 带 来 的 影响 : 由 于 在 独立 区 域 进行 改动 ， 因 此 ， 由 此 带 来 的 影 
一 个 或 最 多 几 个 区 域 中 。 
4、 隐 含 顺序 : 如 果 程 序 通 常 先 从 用 户 那 里 读 取 数据 ， 然 后 再 从 一 个 文件 中 
据 ， 在 设计 系统 时 编写 一 个 函数 ， 隐 含 哪 一 个 首先 执行 的 信息 。 







































































SN 





























响 也 只 限于 





读 取 辅 助 数 


5、 改 进 性 能 : 把 代码 段 放 入 函数 也 使 得 用 更 快 的 算法 或 执行 更 快 的 语言 《如 汇编 ) 来 




















改进 这 段 代 码 的 工作 变 得 容易 些 。 











6、 进 行 集中 控制 :专门 化 的 函数 去 读 取 和 改变 内 部 数据 内 容 ， 也 是 一 种 集 

















式 。 
7、 隐 含 数据 结构 ， 可 以 把 数据 结构 的 实现 细节 隐 含 起 来 。 



































中 的 控制 形 














8、 隐 含 指针 操作 : 指针 操作 可 读 性 很 差 ， 而 且 很 容易 引发 错误 。 通 过 把 它们 独立 在 函 






































数 中 ， 可 以 把 注意 力 集中 到 操作 意图 而 不 是 集中 到 的 指针 操作 本 身 。 
9、 隐 含 全 局 变量 : 参数 传递 。 






































C 语言 中 ， 函 数 其 实 就 是 一 些 语 句 的 的 集合 ， 而 语句 又 是 由 关键 字 和 符号 等 元 素 组 成 ， 


























如 果 我 们 把 关键 字 、 符 号 等 基本 元 素 弄 明白 了 ,函数 不 就 没有 问题 了 么 ? 我 看 未 















































必 。 真正 要 




















编写 出 高 质量 的 函数 来 , 是 非常 不 容易 的 。 前 辈 们 经 过 大 量 的 探讨 和 研究 总 结 出 来 一 下 一 些 











通用 的 规则 和 建议 : 





6. 2， 编 码 风 格 








很 多 人 不 重视 这 点 ,认为 无 所 谓 ， 甚 至 国内 的 绝 大 多 数 教材 也 不 讨论 这 个 话 





题 ， 导 致 学 


LT 


















































E 入 公司 后 仍 要 进行 编码 风格 的 教育 。 我 接触 过 很 多 学 生 , 发 现 他 们 由 于 平时 缺乏 这 种 意识 ， 


养 成 了 不 好 的 习惯 ， 导 致 很 难 改正 过 来 。 代 码 没有 注释 ， 变 量 、 函 数 等 命名 混乱 ， 过 两 天 


[mn 















































都 看 不 懂 自 己 的 代码 。 下 面 是 一 些 我 见 过 的 比较 好 的 做 法 ， 希 望 读者 能 有 所 收获 。 


自 





【规则 6-1]】 每 一 个 函数 都 必须 有 注释 ， 即 使 函数 短 到 可 能 只 有 几 行 。 头 部 说 明 需 要 包 





含 包 含 的 内 容 和 次 序 如 下 : 








JPE eae aae aee eke eak akae ake aeaea ak e ae akee ak eske aeae ae ake kk akae e e ekee ekse ae ake akeke Eak ke kk kakek ekk kkk 


6 Function Name : nucFindThread 

i Create Date : 2000/01/07 

s Author/Corporation : your name/your company name 

* 

a Description ; Find a proper thread in thread array. 

£ If it’s a new then search an empty. 

* 

的 Param ; ThreadNo: someParam description 

a ThreadStatus: someParam description 

* 

* Return Code ; Return Code description,eg: 
ERROR_ Fail: not find a thread 
ERROR_SUCCEED: found 

* 

* Global Variable ; DISP_wuiSegmentAppID 

t File Static Variable : naucThreadNo 

* Function Static Variable ; None 

* 

ポコ ミ ニニ ニニ ニニ ニニ ミニ ニニ ニニ ニニ ミニ ニニ ニニ ニニ ミニ ニニ ニニ ニニ ミニ ニニ ニニ ニニ ー ニ ニー ニー ニー ニニ ニニ ニニ ニニ ニニ ミニミニ ミ ニニ ニニ ニニ ジミ 

Revision History 

a No. Date Revised by Item Description 

* V0.5 2008/01/07 your name ; 
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static unsigned char nucFindThread(unsigned char ThreadNo,unsigned char ThreadStatus) 


{ 
} 








【规则 6-2】 每 个 函数 定义 结束 之 后 以 及 每 个 文件 结束 之 后 都 要 加 一 个 或 若 

















例如 : 











/ ば ボネ ホホ ホホ ポポ ポポ ポポ ポポ ホホ ポポ ポポ ホホ ポポ ホホ ポポ ポポ ポポ ポポ ホホ ポポ ホホ ポポ ホホ ポポ ネネ ポポ ホホ ポポ ポポ ホホ ポポ ポポ ポポ ホホ ポポ ホホ ホホ ポポ ポポ ポポ ネネ ポポ 
* 


* Function1 Description 
* 


ポポ ポポ ホホ ポポ ホホ ポポ ポポ ネ ホホ ポポ ホネ ポポ ポポ ポポ ネ ポポ ボネ ポポ ホホ ホホ ポ ホホ ポ ホネ ポポ ホホ ホホ ポポ ホホ ポポ ホホ ホホ ポポ ネ ホ ポポ ネネ ポポ ホホ ホネ ポポ ポポ ポポ ネ ポポ ホ ポ / 


void Function1(...... ) 


{ 


} 
/Blank Line 


/ ば ポポ ホホ ホホ ポポ ホホ ポポ ポポ ホホ ホネ ホホ ホホ ポポ ポポ ポポ ポポ ポポ ホホ ホホ ポポ ポポ ホ ポ ネネ ポポ ホホ ホホ ホホ ホホ ポポ ポポ ホホ ポポ ホホ ホホ ポポ ポポ ホネ ホネ ポポ ネネ ポポ 
* 


* Function2 Description 
* 


ボネ ポポ ホホ ポポ ポポ ホホ ポポ ポポ ポポ ポポ ポポ ホホ ホホ ポポ ポポ ホホ ポポ ポポ ホホ ポポ ホホ ポポ ホホ ポポ ポポ ホネ ポポ ボネ ホホ ホホ ポポ ポポ ポポ ポポ ホホ ホネ ホホ ホホ ホネ ポポ ポポ ホネ / 


void Function2(...... ) 


{ 


} 
/Blank Line 


/ ド ポポ ホホ ポポ ポポ ホホ ポキ ホネ ポポ ポポ ポポ ポキ ホホ ポポ ホホ ポポ ホホ ポポ ボネ ホホ ボネ ホホ ポポ ホネ ホホ ポポ ポポ ポポ ポポ ポポ ホホ ポポ ネ ポポ ホネ ポポ ポポ ポポ ポポ ホホ ホネ ホホ ポポ ポポ 
* 


* Function3 Description 














a 


aaka aese akafe se ake aleae ake a afe akeke ake Aake ake se ake ae she afe aee ake e ae ale ake afe seale ahe kakak ae aeae akeke afe ake ake ale sea afe ske ake akee akekee aeaea ae akae kakak kakak / 


void Function3(...... ) 


{ 


) 
/Blank Line 








【规则 6-3】 在 一 个 函数 体内 ， 变 量 定义 与 函数 语句 之 间 要 加 空 行 。 
例如 : 








/ さ ボネ ホホ ホホ ポポ ホホ ポポ ポポ ホホ ポポ ホホ ポポ ホホ ポポ ポポ ポポ ポポ ホホ ポポ ホホ ポポ ホホ ポポ ネネ ポポ ホホ ポポ ポポ ホホ ホネ ポポ ポポ ホホ ポポ ホホ ホホ ポポ ポポ ポポ ネネ ポポ 
* 


ネネ ポポ ホホ ポポ ポポ ホホ ポポ ポポ ポポ ポポ ポポ ホホ ホホ ポポ ホホ ポポ ネネ ポポ ホホ ポポ ホホ ホホ ホネ ポポ ポポ ホネ ポポ ホネ ホホ ホホ ポポ ポポ ポポ ネネ ポポ ホホ ポポ ホホ ホホ ポポ ポポ ホネ / 


void Function1() 


{ 
int n; 
/Blank Line 
statementl 
































【 規則 6-4】 ZH LEHRE ij Jae T, EN MEITA 
例如 : 











//Blank Line 
while (condition) 
{ 
statementl: 
/Blank Line 
if (condition) 
{ 
statement2: 
} 
else 
{ 
statement3; 
} 
//Blank Line 
statement4 
} 
































【规则 6-5】 复 条 的 函数 中 ， 在 分 支 语句 ， 循 环 语句 结束 之 后 需要 适当 的 注释 ， 方 便 
分 各 分 文 或 循环 体 














> 





while (condition) 


{ 


statementl: 


(condition) 


{ 
for(condition) 
{ 
Statement2; 
}//end “for(condition)” 
} 
else 























statement3: 
end if (condition)” 


statement4 
W/end “while (condition)” 











【规则 6-6】 修改 别人 代码 的 时 候 不 要 轻易 删除 别人 的 代码 ， 应 该 用 适当 的 注释 方式 ， 
例如 : 











while (condition) 


{ 


statementl : 


//////////777//77////////////77777777/ 
/your name , 2008/01/07 delete 
//if (condition) 

IH 

// for(condition) 

7/ í 

// Statement2; 


// statement3; 
//} 
HAMW/ 


AM 
/ your name , 2000/01/07 add 


new code 
//777//////77/////77//////77/////77777/ 


statement4 















































【规则 6-7】 用 缩 行 显示 程序 结构 ， 使 排版 整齐 ， 缩 进 量 统一 使 用 4 个 字符 〈 不 使 用 TAB 
缩 进 )。 


每 个 编辑 器 的 TAB 键 定 义 的 空格 数 不 一 致 ， 可 能 导致 在 别 的 编辑 器 打开 你 的 代码 乱 成 一 
Hr. 

【规则 6-8】 在 函数 体 的 开始 、 结 构 / 联 合 的 定义 、 枚 举 的 定义 以 及 循环 、 判 断 等 语句 中 
的 代码 都 要 采用 缩 行 。 


【规则 6-9】 同 层次 的 代码 在 同 层次 的 缩 进 层 上 。 
例如 : 


提倡 的 的 风格 不 提倡 的 风格 


void Function(int x) void Function(int x) 


{ { 































































































/program code /program code 


} 


struct tagMyStruct struct tagMyStruct 





if (condition) if (condition){ 


{ //program code 


/program code }else{ 


//program code 


} 


//program code 











【规则 6-10】 代 码 行 最 大 长 度 宜 控制 在 80 个 字符 以 内 ， 较 长 的 语句 、 表 达 式 等 要 分 成 
多 行书 写 。 


【规则 6-11】 长 表达 式 要 在 低 优先 级 操作 符 处 划分 新 行 ， 操 作 符 放 在 新 行 之 首 《〈 以 便 突 
出 操作 符 )。 拆 分 出 的 新 行 要 进行 适当 的 缩 进 ， 使 排版 整齐 ， 语 句 可 读 。 
例如 : 


if ((very_longer_variablel >= Very_longer_Variable12) 
&& (very_longer_variable3 <= very_longer_variable14) 
&& (very_longer_variableS <= very_longer_variable16)) 





















































{ 
} 


dosomething(); 


for (very_longer_initialization; 
Very_longer_condition: 
Very_longer_update) 

{ 


} 


dosomething(): 














【规则 6-12】 如 果 函 数 中 的 参数 较 长 ， 则 要 进行 适当 的 划分 。 
例如 : 


void function(float very_longer_var1, 
float very_longer_var2, 
float very_longer_var3) 





























【规则 6-13】 用 正确 的 反义词 组 命名 具有 互 斥 意义 的 变量 或 相反 动作 的 函数 等 。 
例 如 : 


int aiMin Value; 














int aiMax Value; 
int niSet_Value(…): 
int niet_Value(): 








【规则 6-14】 如 果 代 码 行 中 的 运算 符 比 较 多 ， 用 括号 确定 表达 式 的 操作 顺序 ， 避 免 使 用 
默认 的 优先 级 。 





例如 : 
leap_year = ((year % 4 == 0) && (year % 100 != 0)) || (year % 400 == 0); 


【规则 6-15】 不 要 编写 太 复 杂 的 复合 表达 式 。 
例如 : 
i=a>=b&&c<d&&c+f<=g+h; 复合 表达 式 过 于 复杂 


【规则 6-16】 不 要 有 多 用 途 的 复合 表达 式 。 
例如 : 
d=(a=b+c)+r; 


该 表达 式 既 求 a 值 又 求 d 值 。 应 该 拆 分 为 两 个 独立 的 语句 : 



























































a=b+c; 
d=a+tr; 
【建议 6-17】 尽 量 避 免 含 有 否定 运算 的 条 件 表 达 式 。 
例如 : 


如 : if (1@mum >= 10)) 
应 改 为 : if (num < 10) 














【规则 6-18】 参 数 的 书写 要 完整 ， 不 要 贪图 省 事 只 写 参 数 的 类 型 而 省 略 参数 名 字 。 如 果 
函数 没有 参数 ， 则 用 void 填充 。 
例如 : 











提倡 的 风格 不 提倡 的 风格 





void set_value(int width, int height); Void set_value (int, int); 
float get_value(void): float get_value (); 


6. 2, 函数 设计 的 一 般 原 则 和 技巧 


如 























【规则 6-19】 原 则 上 尽量 少 使 用 全 局 变量 ， 因 为 全 局 变量 的 生命 周期 太 长 ， 容 易 出 错 ， 
也 会 长 时 间 占 用 空间 . 各 个 源 文件 负责 本 身 文件 的 全 局 变量 , 同时 提供 一 对 对 外 函数 , 方 
便 其 它 函 数 使 用 该 函数 来 访问 变量 。 比 如 : niSet_ValueName(…: met_ValueName(: 
不 要 直接 读 写 全 局 变量 , 尤其 是 在 多 线程 编程 时 ， 必 须 使 用 这 种 方式 ， 并且 对 读 写 操作 
加 锁 。 


【规则 6-20】 参 数 命名 要 恰当 ， 顺 序 要 合理 。 
例如 编写 字符 串 找 贝 函数 str_copy， 它 有 两 个 参数 。 如 果 把 参数 名 字 起 为 strl 和 str2， 例 




















































































































void str_copy (char *str1, char *str2); 
那么 我 们 很 难 搞 清 楚 究 竟 是 把 strl 拷贝 到 str2 中 ， 还 是 刚好 倒 过 来 。 

可 以 把 参数 名 字 起 得 更 有 意义 ， 如 叫 strSource 和 strDestination。 这 样 从 名 字 上 就 可 
以 看 出 应 该 把 strSource 拷贝 到 sttDestination 。 

还 有 一 个 问题 ,这 两 个 参数 那 一 个 该 在 前 那 一 个 该 在 后 ?参数 的 顺序 要 遵循 程序 员 
的 习惯 。 一 般 地 ， 应 将 目的 参数 放 在 前 面 ， 源 参数 放 在 后 面 。 












































如 果 将 函数 声明 为 : 

Void str_copy (char *strSource, char *strDestination); 
别人 在 使 用 时 可 能 会 不 假 思 索 地 写成 如 下 形式 : 

char str[20]; 

str_copy (str, “Hello World”); 参数 顺序 颠倒 























【规则 6-21】 如 果 参 数 是 指针 ， 且 仅 作 输入 参数 用 ， 则 应 在 类 型 前 加 const， 以 防止 该 
指针 在 函数 体内 被 意外 修改 。 
例如 : 


void str_copy (char *strDestination, const char *strSource): 


=E 


【规则 6-22】 不 要 省 略 返 回 值 的 类 型 ， 如 果 函 数 没 有 返回 值 ， 那 么 应 声明 为 void 类 型 。 
如 果 没 有 返回 值 ， 编 译 器 则 默认 为 函数 的 返回 值 是 int 类 型 的 。 
【规则 6-23】 在 函数 体 的 “入 口 处 ”， 对 参数 的 有 效 性 进行 检查 。 尤 其 是 指针 参数 ， 尽 
使 用 assert 宏 做 入 口 校 验 , 而 不 使 用 这 语句 校 验 。( 关 于 此 问题 讨论 , 详 见 指针 与 数组 那 章 。) 



















































































la 








【 規則 6-24】 return 语句 不 可 返回 指向 “ 栈 内 存 ” 的 “指针 ”， 因 为 该 内 存在 函数 体 结 
束 时 被 自动 销毁 。 例 如 ; 
char * Func(void) 


( 








char str[30]: 


return str; 


) 
str 属于 局 部 变量 , 位 于 栈 内 存 中 , 在 Func 结束 的 时 候 被 释放 , 所 以 返回 str 将 导致 错误 。 












































【规则 6-25】 函 数 的 功能 要 单一 ， 不 要 设计 多 用 途 的 函数 。 微 软 的 Win32 API 就 是 违反 
本 规则 的 典型 ， 其 函数 往往 因为 参数 不 一 样 而 功能 不 一 ， 导 致 很 多 初学 者 迷惑 。 

【规则 6-26】 函 数 体 的 规模 要 小 ， 尽 量 控制 在 80 行 代码 之 内 。 

【建议 6-27】 相 同 的 输入 应 当 产 生 相 同 的 输出 。 尽 量 避 人 免 函 数 带 有 “记忆 ”功能 。 
带 有 “记忆 ”功能 的 函数 ， 其 行为 可 能 是 不 可 预测 的 ， 因 为 它 的 行为 可 能 取决 于 某 种 
“记忆 状态 “。 这样 的 函数 既 不 易 理 解 义 不 利于 测试 和 维护 。 在 C 语言 中 ， 函 数 的 static 
局 部 变量 是 函数 的 “记忆 ”存储 器 。 建 议 尽量 少 用 static 局 部 变量 ， 除 非 必 需 。 
【建议 6-28】 避 免 函 数 有 太 多 的 参数 ， 参 数 个 数 尽量 控制 在 4 个 或 4 个 以 内 。 如 果 参 数 太 
多 ， 在 使 用 时 容易 将 参数 类 型 或 顺序 搞 错 。 微 软 的 Win32 API 就 是 违反 本 规则 的 典型 ， 
其 函数 的 参数 往往 七 八 个 甚至 十 余 个 。 比 如 一 个 CreateWindow 函 数 的 参数 就 达 11 个 之 
多 。 
【建议 6-29】 尽 量 不 要 使 用 类 型 和 数目 不 确定 的 参数 。 


C 标准 库 函 数 printf 是 采用 不 确定 参数 的 典型 代表 ， 其 原型 为 : 
int printf(const chat *format[, argument]-); 
这 种 风格 的 函数 在 编译 时 丧失 了 严格 的 类 型 安全 检查 。 
【建议 6-30】 有 时 候 函 数 不 需 要 返回 值 ， 但 为 了 增加 灵活 性 如 支持 链 式 表达 ， 可 以 附加 
返回 值 。 例 如 字符 串 拷贝 函数 strcpy 的 原型 : 


























































































































































































































a 


char *strcpy(char *strDest, const char *strSrc): 


strcpy KZK strSre 拷贝 至 输出 参数 strDest 中 ， 同 时 函数 的 返 
并 非 多 此 一 举 ， 可 以 获得 如 下 灵活 性 : 


char str[20]; 


回 值 又 是 strDest。 这 样 做 




















int length = strlen(strcpy(str, “Hello World”) ): 


に っ 























【建议 6-31) 不 仅 要 检查 输入 参数 的 有 效 性 ， 还 要 检查 通过 其 它 途 径 进 入 函数 体内 的 变 











量 的 有 效 性 


【规则 6-32】 函 














getchar žr, 


会 拿 这 个 例子 来 警示 1 


char c; 

c = getchar(): 
if(EOF == c) 
{ 


) 


按照 getchar 名 字 的 意思 , 








， 例 如 全 
数 名 与 返回 


违反 这 条 规则 的 典型 代表 训 
为 它 实 在 太 经 典 ， 太 容易 让 人 犯错 误 了 。 所 以 ， 每 一 个 有 经 验 的 作者 都 
也 的 读者 ， 我 这 里 也 是 如 此 : 























局 变量 、 文 件 句柄 等 。 


值 类 型 在 语义 上 不 可 冲突 。 




















值 却 是 int 类 型 





プ マ ご ビッ 








FE] 





int getchar(void); 








Hd 








6. 4， 函 数 递归 


JIR 





c 是 char 类 型 的 , 取 值 范围 
EOF 的 值 将 无 法 全 部 保存 到 c 内 ， 会 发 生 
句 有 可 能 总 是 失败 。 这 种 潜在 的 危险 ， 如 果 不 是 犯 过 一 次 错 ， 肯 怕 很 x 





是 C 语 言 标准 库 函 数 getchar。 几 平 没有 一 部 名 著 没 有 提 到 









































á 














应 该 将 变量 c 定义 为 char 类 型 。 BERDE, getchar 函数 的 


型 为 : 








=, 














是 [-128,127], 如 果 宏 EOF 的 值 在 char 的 取 值 范围 之 外 ， 
RE, 次 EOF 值 的 低 8 位 保存 到 c 里 。 这 样 让 语 
ERIL o 

































































6. 4. 1， 一 个 简单 但 易 出 错 的 递归 例子 





几乎 每 





方 犯 销 














Void fun(int i) 


{ 


if (1>0) 
{ 

fun(i/2 
} 


); 


printf("%d\n",i); 


int main() 





À C 语言 基础 的 书 都 讲 到 了 函数 递归 的 问题 ， 但 是 初学 者 仍然 容易 在 这 个 地 
误 。 先 看 看 下 面 的 例子 : 





fun(10); 


return 0; 


) 
问 : 输出 结果 是 什么 ? 


这 是 我 上 课时 ， 一 个 学 生 问 我 的 问题 。 他 不 明白 为 什么 输出 的 结果 会 是 这 样 : 
0 








1 
2 


10 
他 认为 应 该 输出 0. 因 为 当 i 小 于 或 等 于 0 时 递归 调用 结束 , 然后 执行 printf 函数 打印 i 的 值 。 
这 就 是 典型 的 没 明白 什么 是 递归 。 其 实 很 简单 ，printft"gdwm",i; 语 句 是 fun 函数 的 一 部 
分 ， 肯 定 执行 一 次 fun 函数 ， 就 要 打印 一 行 。 怎 么 可 能 只 打印 一 次 呢 ? 关键 就 是 不 明白 怎么 
展开 递归 函数 。 展 开 过 程 如 下 
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void fun(int i) 


( 
if (i>0) 
( 
/fun(i/2); 
ifG/2>0) 
( 
ifG/4>0) 
( 
} 
printf("%d\n",i/4); 
} 
printf("%d\n",i/2); 
} 
printf("%d\n",i); 
} 
























































这 样 一 展开 , 是 不 是 清晰 多 了 ? 其 实 递归 本 身 并 没有 什么 难处 , 关键 是 其 展开 过 程 别 弄 错 了 。 























6. 4. 2， 不 使 用 任何 变量 编写 strlen 函数 




















看 到 这 里 ， 也 许 有 人 会 说 ，strlen 函数 这 么 简单 ， 有 什么 好 讨论 的 。 是 的 ， 我 相信 你 能 
熟练 应 用 这 个 函数 ， 也 相信 你 能 轻易 的 写 出 这 个 函数 。 但 是 如 果 我 把 要 求 提高 一 些 呢 : 


不 允许 调用 库 函 数 ， 也 不 允许 使 用 任何 全 局 或 局 部 变量 编写 int my_strlen (char *strDest); 




































































似乎 问题 就 没有 那么 简单 了 吧 ? 这 个 问题 曾经 在 网 络 上 讨论 的 比较 热烈 ， 我 几乎 是 全 
程 “ 观 战 ” 差点 也 忍 不 住 手 痒 了 。 不 过 因为 我 的 解决 办 法 在 我 看 到 帖子 时 已 经 有 人 提出 了 ， 
所 以 作罢 。 
解决 这 个 问题 的 办 法 由 好 几 种 ， 比 如 网 套 有 编 语 言 。 因 为 嵌 套 汇编 一 般 只 在 能 入 式 底 
层 开发 中 用 到 ， 所 以 本 书 就 不 打算 讨论 C 语言 谋 套 汇编 的 知识 了 。 有 兴趣 的 读者 ， 可 以 查 
找 相关 资料 。 

也 许 有 的 读者 想到 了 用 递归 函数 来 解决 这 个 问题 。 是 的 ， 你 应 该 想 
个 问题 放 在 讲解 函数 递归 的 时 候 讨论 ,既然 已 经 有 了 思路 , 这 个 问题 就 很 
int my_strlen( const char* strDest ) 


{ 












































































































































得 到 ， 因 为 我 把 这 
简单 了 。 代码 如 下 : 



































assert(NULL != strDest): 
if (\0' == *strDest) 
( 


return 0: 


else 


return (1 + my_ strlen(++strDest)); 


は y = 


一 步 : 用 assert 宏 做 入 口 校 验 。 
二 步 : 确定 参数 传递 过 来 的 地 址 上 的 内 存 存储 的 是 否 为 \0'。 如 果 是 ， 表 明 这 是 一 个 
或 者 是 字符 串 的 结束 标志 。 
I 果 参 数 传递 过 来 的 地 址 上 的 内 存 不 为 \0'， 则 说 明 这 个 地 址 上 的 内 存 上 存储 
的 是 一 个 字符 。 既 然 这 个 地 址 上 存储 了 一 个 字符 ， 那 就 计数 为 1， oR 
类 型 元 素 的 大 小 ， 然 后 再 调用 函数 本 身 。 如 此 循环 ， 当 地 址 加 到 字符 串 的 结束 标志 符 \O' 时 ， 
递归 停止 

当然 ， 同 样 是 利用 递归 ， 还 有 人 写 出 了 更 加 简洁 的 代码 : 
int my_strlen( const char* StrDest ) 
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dn 
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return *strDest?1+strlen(strDest+1):0; 

) 

这 里 很 巧妙 的 利用 了 问号 表达 式 , 但 是 没有 做 参数 入 口 校 验 , 同時 用 *strDest RREA 
== *strDest) 也 不 是 很 好 。 所 以 ， 这 种 写法 虽然 很 简洁 ， 但 不 符合 我 们 前 面 所 讲 的 编码 规范 。 
可 以 改写 一 下 : 

int my_strlen( const char* strDest ) 


{ 






















































































assert(NULL != strDest); 
return (\0' に *strDest)?(1+my_strlen(strDest+1)):0; 
} 
上 面 的 问题 利用 函数 递归 的 特性 就 轻易 的 搞定 了 , 也 就 是 说 每 调用 一 遍 my_strlen 函数 ， 
其 实 只 判断 了 一 个 字 节 上 的 内 容 。 但 是 ， 如 果 传 入 的 字符 串 很 长 的 话 ， 就 需要 连续 多 次 函数 
调用 ， 而 函数 调用 的 开销 比 循环 来 说 要 大 得 多 ， 所 以 ， 递 归 的 效率 很 低 ， 递 归 的 深度 太 大 其 
至 可 能 出 现 错误 比如 栈 溢出 )。 所 以 ， 平 时 写 代码 ， 不 到 万 不 得 已 ， 尽 量 不 要 用 递归 。 即 































































































便 是 要 用 递归 ， 也 要 注意 递归 的 层次 不 要 太 深 ， 防 1 


E 确 ， 否 则 ， 递 归 可 能 没完 没 了 。 





件 一 定 要 了 


H H 








BILERA H JERR; 同时 递归 的 停 J 
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第 七 章 ”文件 结构 























一 个 工程 是 往往 由 多 个 文件 组 成 。 这 些 文件 怎么 管理 、 怎 么 命名 都 是 非常 重要 的 。 下 面 
给 出 一 些 基 本 的 方法 ， 比 较 好 的 管理 这 些 文件 ， 避 免 错 误 的 发 生 。 




































































7.1， 文 件 内 容 的 一 般 规 则 


【规则 7-1】 每 个 头 文件 和 源 文 件 的 头 部 必须 包含 文件 头 部 说 明和 修改 记录 。 
源 文件 和 头 文件 的 头 部 说 明 必 须 包含 的 内 容 和 次 序 如 下 : 
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* File Name ; FN_FileName.c/ FN_FileName.h 

* Copyright Š 2003-2008 XXXX Corporation, All Rights Reserved. 
* Module Name : Draw Engine/Display 

* CPU : ARM 

* RTOS : Tron 

* 

* Create Date : 2008/10/01 

* Author/Corporation : 。 WhoAml/your company name 

* Abstract Description : Place some description here. 

* 

E----- Revision History--------------------------------- 

* No Version Date Revised By Item Description 

* 1 V0.95 08.05.18 WhoAml abcdefghijklm WhatUDo 

x 

に いと と と と と と と と と と と と と と と と と と と と と と と と と と と と と と と と と と と と と 














【规则 7-2】 各 个 源 文件 必须 有 一 个 头 文件 说 明 ， 头 文件 各 部 分 的 书写 顺序 下 : 








其 中 Multi-Include-Prevent Section 是 用 来 防止 头 文件 被 重复 包含 的 。 

如 下 例 : 

#ifndef _ FN_FILENAME_H 
#define _ FN_FILENAME_H 
#endif 

其 中 “FN_FILENAME” 一 般 为 本 头 文件 名 大 写 , 这 样 可 以 有 效 避 免 重复 ,因为 同一 工程 
中 不 可 能 存在 两 个 同名 的 头 文件 。 
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2003-2008 XXXX Corporation, All Rights Reserved. 


Draw Engine/Display 


FN_FileName.h 


Copyright 
Module Name 


File Name 


Tron 


ARM7 


2008/10/01 


Create Date 
Author/Corporation 


WhoAml/your company name 
Place some description here. 


Abstract Description 


----------------------------------------Revision History--------------------------------- 


Description 


Item 
abcdefghijklm 


Revised By 


08.05.18 WhoAmI 


Date 


No Version 


WhatUDo 


1 


Multi-Include-Prevent Section 


* 


FN_FILENAME_H 


#ifndef 


FN FILENAME H 


#define 


Debug switch Section 


#define D_DISP BASE 


Include File Section 


* 


#include "IncFile.h" 


Macro Define Section 


(4) 


#define MAX_TIMER_OUT 


Define Section 


Struct 
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Prototype Declare Section 








unsigned int MD_guiGetScanTimes(void); 


#endif 


【规则 7-3】 源 文件 各 部 分 的 书写 顺序 如 下 : 


| 
| 
> | 








* 。 Hile Name 3 FN_FHileName.c 

* Copyright ; 2003-2008 XXXX Corporation, All Rights Reserved. 
* Module Name ; Draw Engine/Display 

* CPU : ARM7 

* 。 RTOS | Tron 

* Create Date : 2003/10/01 

* Author/Corporation : WhoAml/your company name 

* Abstract Description : Place some description here. 

* 

X ---------------------- Revision History--------------------------------- 

* No Version Date Revised By Item Description 

* 1 V0.95 00.05.18 WhoAml abcdefghijklm WhatUDo 
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Debug switch Section 
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#define D DISP_BASE 


#include "IncFile.h" 
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ia Macro Define Section 
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#define MAX TIMER OUT (4) 
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* Struct Define Section 
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typedef struct CM_RadiahonDose 


{ 
unsigned char ucCtgID; 


char cPatId_a[MAX_PATI_LEN]; 
}CM_RadiationDose_st, *pCM_RadiationDose_st; 


s Prototype Declare Section 
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unsigned int MD_guiGetScanTimes(void); 


EEEE EEEE EEEE EEEE EEEE EEEE EE E E EE E E EE EE EE E koko ok 


* Global Variable Declare Section 
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extern unsigned int MD_guiHoldBreathStatus: 


ネ File Static Variable Define Section 
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static unsigned int nu1NaviSysStatus: 











PF 


【规则 7-4】 需 要 对 外 公开 的 常量 放 在 头 文件 中 ， 不 需要 对 外 公开 的 常量 放 在 定义 文件 
的 头 部 。 





ffin 








7.2， 文 件 名 命名 的 规则 











【规则 7-5】 文 件 标识 符 分 为 两 部 分 ， 即 文件 名 前 绥 和 后 级 。 文 件 名 前 缀 的 最 前 面 要 使 
用 范围 限定 符 一 一 模块 名 《文件 名 ) 缩写 ; 


【规则 7-6】 采 用 小 写字 母 命名 文件 , 避免 使 用 一 些 比较 通俗 的 文件 名 , 如 : public. c 等 。 














